热搜词
发表于 2017-10-7 23:07:11 | 显示全部楼层 |阅读模式

原文作者Asha Laxmi在文章Using CSS’s object-fit and object-position Properties中介绍了两个CSS属性:object-fit和object-position,有助于我们解决images和videos这类可置换元素在布局时遇到的尺寸和宽高比问题。在原来的工作中我也遇到过这种情况,如果图像作为内容出现,考虑到前端倡导的内容和样式分离,那么把图片设置成一个div的背景是不合适的。看了这篇文章才知道,利用object-fit和object-position为img标签书写css样式,可以起到类似于background-size和background-position的作用,很好地解决了这个问题。翻译出来和大家共享。


以下是正文:

在给一个网站布局时,视频和图像很容易出现问题。

假设你在CSS里明确指定了一个图像的宽度。如果这个宽度是使用百分比或者视口单位(viewport units)指定的,同时图像的宽度设置成了auto,那么调整浏览器窗口的时候图像的宽高比不变。

然而有时候,可能你可用的空间有限,需要限制图像的高度。这种情况下调整浏览器窗口肯定会破坏图像的宽高比。

很多关于图像尺寸和宽高比的问题,包括我刚才提到的,可以使用CSS的 object-fitobject-position 属性来解决。

这两个属性和我们熟悉的background-size和background-position属性比较类似。所以,即使你从没听说过,理解起来也不会很困难。

这两个属性都对可置换元素(replaced elements)起作用。然而,为了简洁起见,在这篇文章中,我们用images举例说明。

为什么使用 object-fit 和 object-position?

你可能想知道明明可以使用background-size 和 background-position,为什么还要用这两个属性。事实上,至少有两个很好的理由。

首先,考虑这样一个场景,你有一张图片,它是一篇文章的一部分。如果图片的尺寸像之前描述的那样进行设置,调整浏览器窗口将会破坏它的宽高比。这种情况下,仅使用一个img标签是不够的,你需要使用一个带有background-size 和 background-position属性的div。

这种解决方案的问题就是你必须始终保证内容和表现分离,而background属性用于表现的。当你的图像实际上是内容的一部分时,使用object-fit和object-position以及img标签来实现相同的结果会更有意义。

另外一个理由是background属性不能应用于videos,但是object-fit和object-position可以。因此,当显示视频的时候,object-fit和object-position是你唯一的选择。

object-fit 属性

这个属性决定了像images和videos这样的可替换内容占据其content box空间的方式。它有5个可能的值:

  • fill

  • contain

  • cover

  • none

  • scale-down


使用这个属性的语法如下:

.fit-image {  object-fit: fill|contain|cover|none|scale-down;}

当设置成fill时,图像的大小会正好填满content box。这种情况下的图片高度等于content box本身的高度。这也是object-fit属性的初始值。

CodePen上查看SitePoint (@SitePoint) 的笔记Object Fit: Fill

contain重设图像的尺寸,在保持宽高比的情况下使其填充到content box中。这有两个目的,既避免了图像失真,又能保证图像在content box里。如果图片不能完全覆盖content box,背景色会填满剩余的空间。

CodePen上查看SitePoint (@SitePoint) 的笔记Object Fit: Contain

contain适用的场景是,当你不知道图像的尺寸却仍想把它放到一个已知宽度的容器内时。

如果你想要在保证宽高比的情况下让图像完全填满它的content box,你应该用 cover属性。这种情况下,图片会缩放到较短边恰好适合containing box。溢出盒子的图像部分会被裁掉。

CodePen上查看SitePoint (@SitePoint) 的笔记Object Fit: Cover

要使用图像本身的尺寸显示,忽略containing box设置的宽高值,你可以使用none。这种情况下的图像不会被重新设置大小。然而,如果图像的任何部分超出了它的containing box就会被裁掉。

CodePen上查看SitePoint (@SitePoint) 的笔记Object Fit: None

object-fit属性的最后一个值是scale-down。顾名思义,即按比例缩小图像,就像被设置成none或者contain,得到一个较小的图像。换句话说,如果我们图像的尺寸都小于其各自的containing box的尺寸,那么就会设置成none。

CodePen上查看SitePoint (@SitePoint) 的笔记Object Fit: scale-downobject-position属性

正如前面看到的,cover属性会填满containing box的所有空间,并保持图像原有的宽高比。图像也会默认居中显示。这适用于聚焦点在中心的情况。但是,如果图像的聚焦点不在中心怎么办?

object-position属性便可以帮上忙。它的工作原理就像background-position。下面的代码段将图像的位置设成了左上角:

.zero-zero {  object-position: 0px 0px;}

定义时可以用像素,也可以用百分比。例如,object-position: 0% 0%是左上角,object-position: 100% 100%是右下角。当你不知道图像的尺寸时,这样写很方便。

当图像的宽高比和containing box的宽高比很接近时,设置object-position似乎不会产生多大影响。然而,当宽高比差别很显著时,结果就会变得明显。下面的CodePen demo说明了我的观点。

CodePen上查看SitePoint (@SitePoint) 的笔记GZewMJ

在第二个例子中设置位置为左上角,可以聚焦到太阳上。

值得一提的是,object-position属性是可写入动画的,运用得当的话可以产生惊人的效果。考虑下面的代码:

img {  /* other CSS here... */  object-fit: cover;  object-position: 0% 0%;  animation: ltr 5s alternate infinite;}  @keyframes ltr {  0% {    object-position: 0% 0%;  }  25% {    object-position: 20% 0%;  }  100% {    object-position: 100% 100%;  }}

这段代码里,我使用keyframe为图像位置做动画,如你所见,最终结果看起来很棒。

CodePen上查看SitePoint (@SitePoint) 的笔记Animating the object-position Property

这也可以应用到videos中,当焦点从一个人移到另一个人上时。

跨浏览器支持和兼容性

尽管这些属性经证明是非常有用的,但你是否能真正使用它们取决于你的浏览器是否支持。object-fit可以被除了IE/Edge以外的所有主流浏览器支持,object-position可以被除了IE/Edge和Safari以外的所有主流浏览器支持。

一般来说,你可能恰好需要object-fit属性。如果你愿意牺牲一部分人的用户体验,那么你可以应用这个属性。如果你需要兼容旧的浏览器,可以试一试the polyfill by Federico Brigante

结论

希望你们喜欢这篇教程和附带的演示。如果我遗漏了什么或者你想要补充些什么,可以给我留言。


全部评论0
回复
您需要登录后才可以回帖 登录 | 注册

本版积分规则

QQ|Archiver|手机版|小黑屋|管理员之家 ( 苏ICP备2023053177号-2 )

GMT+8, 2025-1-16 02:55 , Processed in 0.199244 second(s), 22 queries .

Powered by Discuz! X3.5

Cpoyright © 2001-2025 Discuz! Team