CSS Essential

What is CSS?


The basic goal of the Cascading Stylesheet (CSS) language is to allow a browser engine to paint elements of the page with specific features, like colors, positioning, or decorations.

CSS主要作用是设定元素(通常称为盒子)的属性,并将设定这些元素在页面的位置关系。CSS解决的主要问题是:

  1. 如何选择指定的元素?选择器(Selector)
  2. 如何设定元素的属性?盒子模型(Box Model)
  3. 如何设置元素间的位置关系?定位机制(Positioning Scheme)

本文主要介绍CSS的基本用法,包括如何使用选择器,如何设定样式以及如何布局定位,这些涉及到选择器、盒子模型、普通流、可视化格式模型和块格式化环境等核心概念,理解这些核心概念是熟练运用CSS的基础。

若要详细了解相关内容,可参考文中链接,以及MDNW3w3school关于CSS的详细介绍。

基础知识

基本语法

CCS语法元素主要由选择器,属性和值构成,通过规则给选择器范围内的属性赋值。

CSS语法

CSS语法(规则)

各部分元素含义:

基本选择器

选择器相当于作用域规则,选择属性设置起作用的范围。另一个相关概念是块格式化环境。

常用选择器

E 标签选择器,匹配所有使用E标签的元素。

.info class选择器,匹配所有class属性中包含info的元素。

#footer id选择器,匹配所有id属性等于footer的元素。

id选择器和class选择器的区别:

多选择器组合

E,F 多元素选择器,同时匹配所有E元素或F元素,E和F之间用逗号分隔。

E F 后代元素选择器,匹配所有属于E元素后代的F元素,E和F之间用空格分隔。

E > F 子元素选择器,匹配所有E元素的子元素F。

E + F 毗邻元素选择器,匹配所有紧随E元素之后的同级元素F。

如果两个选择器紧连,表示同时满足两个条件的内容。

<style>
div.content {font-size: 1.2em; line-height: 200%}
div.emphasis {font-weight: bold}
div#lily {color: white; background: black}
</style>
<div class="content emphasis" id="lily">
The flower of lily of the valley is like tinkler, be born at spending cauline top to show raceme.
</div>

上例中,div.emphasis表示套用.emphasis样式的div标签。

优先级别

基本规则是行内样式 > id样式 > class样式 > 标签名样式,也就是,选择越具体优先级越高。如下元素:

<div id="ID" class="CLASS" style="color:black;"></div>

作用在其上样式的优先级从低到高是div < .CLASS < div.CLASS < #ID < div#ID < #ID.CLASS < div#ID.CLASS

继承与覆盖:

子元素没有设置的属性会从父元素继承而来。同一元素的同一属性如有多次设置,优先级高的覆盖优先级低的;若是同一优先级,后设设置覆盖前设置。

<style>
div {font-size: 1em}
div.content {font-size: 1.2em; line-height: 200%}
div.emphasis {font-size: 1.4em; font-weight: bold}
div#lily {color: white; background: black}
</style>
<div class="content emphasis" id="lily">
The flower of lily of the valley is like tinkler, be born at spending cauline top to show raceme.
<div id="water_lily"></div>
</div>

上例中,div设定的font-size优先级太低,不会生效;.emphasis覆盖.contentfont-size#lilyfont-size最终为1.4em#water_lily继承了#lily的所有属性。

属性赋值

确定属性最终取值要经历3步CSS2经历4步):首先获取CSS样式中的指定值(specified value),然后如有必要则转换为绝对值或计算值(computed value),最后根据局部环境的约束再转换为实际值(actual value)。

指定值可能是绝对值(例如:2mmred等),也可能是相对值(例如:auto1.2em12%等)。对于绝对值而言不需要计算即可获得计算值,相对值需要再借助参考值计算获得计算值。如果属性没有指定值,它的取值继承父元素。

在特定的客户端环境,可能还需要将计算值转换为实际值,比如可能需要将小数的边界近似取整。

常用的赋值规则


  1. 颜色赋值的三种形式:关键字、RGB空间、HSL空间;
  2. 长度赋值的两种形式:以emexchremvwvhvminvmax为单位的相对赋值,以cmmminptpcpx为单位的绝对赋值;
  3. 百分数赋值:widthmarginpadding等接受百分数赋值。

示例:

em等单位和百分数都是相对赋值,需要继承父属性的参考值。em是针对字体大小的值,$w$ em = $w$ $\times$ font-size

<div style="font-size:18px;">
  Full size text (18px)
  <span style="font-size:50%;">50%</span>
  <span style="font-size:200%;">200%</span>
</div>

上述代码中,50%从父属性继承而来的值是18px,然后再乘以50%(等价于0.5em),实际大小是9px查看效果)。

当一个属性可以有多个方向可设置值时,存在形如以下简写的赋值规则:

位置赋值规则

1 border-width: 1em

2 border-width: 1em 2em

3 border-width: 1em 2em 3em

4 border-width: 1em 2em 3em 4em

盒子模型

CSS将HTML的元素(可认为是标签)定义为适合CSS处理的矩形盒(rectangular box)。盒子模型(box model)描述了这些矩形盒的尺寸、属性(颜色、背景和边框等)和位置特性,浏览器根据盒子模型实现页面的渲染与显示。

按照HTML的元素是否新开一行可分为块(block-level)元素和内联(inline)元素,这一特性对设置元素的布局至关重要。

HTML的块(block-level)元素和内联(inline)元素


块元素充满父元素的所有空间,每个块元素都会另起一个新行显示。块元素之内还可以包含块元素和行内元素。HTML定义的块元素包括<div>, <span>, <p>等。内联元素只占据标签包含内容的空间,不会另起新行显示。内联元素通常只包含内联元素。HTML定义的内联元素包括<span>, <code>, <textarea>等。

d

上例中,<p>是块元素,新起一行显示,并且撑满了行,<em>是内联元素,紧接上一元素显示。

CSS的display属性可设定HTML元素生成盒子的类型是块还是内联,常用的属性值有block, inline, inline-block, none。将display属性设为block,可将内联元素转为块元素。盒子模型有四类边:margin edge,border edge,padding edge,content edge。

CSS语法

盒子模型的四类边

.box {with: ...; height: ...}设置的是只是Content尺寸。相关边的的设置方法是:

.box {
    width: ...;
    height: ...;
    padding: ...;
    border: ...;
    margin: ...;
}

内联元素设置属性heightwidth是没有用的,致使它变宽变大的原因是内部元素的宽高+padding

满足一定条件俩个盒子的外边届(margin)会叠加,使得估计这俩个盒子的位置关系变得复杂。

外边叠加(margin collapsing)


上边距(top margin)和下边距(bottom margin)有时会叠加,叠加后两者最大值作为两者间的边距。发生外边叠加的3种情况(图解)1:

  1. 兄弟(Adjacent siblings)块:相邻块中,兄的下边界和弟的上边界会叠加;
    1. 父与首尾孩子(Parent and first/last child)块:父块和首个孩子的margin-top之间不被任何东西分隔,则它们会叠加;父块和最后孩子的margin-bottom之间不被任何东西分隔,则它们会叠加;
    2. 空块(Empty blocks):不存在border、padding、inline content、height或min-height分隔块的margin-top和margin-bottom,上下边界会叠加。

    注意事项:

    • 当与负边界叠加时,叠加后的边界是最大正边界和最小边界之和;
    • 浮动的(floating)和绝对定位的(absolutely positioned)元素不参与外边叠加(这是因为创建了新的块格式化环境,而边界叠加只发生在同一块格式化环境)。

可视化格式模型

可视化格式模型解决的主要问题


  1. 如何生成盒子?
  2. 盒子如何在页面布局?

可视化格式模型(Visual formatting model)是用于处理网页文档并显示到虚拟设备上的算法,该模型将文档中的每个元素生成符合盒子模型的盒子,然后对这些模型进行布局。可视化格式模型包括盒子的生成和定位两部分。

生成机制(Box Generation)

生成盒子的类型通过CSS属性display设定,当设定属性为blocklist-itemtable时,生成块盒子,当设定属性为inlineinline-blockinline-table时,生成内联盒子。HTML元素的属性决定了该元素生成盒子类型的默认值,块元素默认生成块盒子,内联元素默认生成内联盒子。不同类型盒子的定位机制不同。

定位机制(Positioning Scheme)

CSS的盒子有3种基本定位机制:普通流(normal flow)、浮动(floats)和绝对定位(absolute position),默认定位机制是普通流。在普通流中,盒子一个接一个排列,floats算法可以将盒子从普通流中抽出来,绝对定位通过包含它盒子的坐标系统来定位。

Normal Flow

如何进入普通流(normal flow)


CCS将盒子的position属性设置为staticrelative,并且将float属性设置为none。默认值position: staticfloat: none

普通流可以理解为盒子按照读入的先后次序依次处理,就像水流一样连续有序,每个页面对应一个流。浮动和绝对定位,可将读入序列中的某些盒子从这个流中抽取出来,单独处理。

在普通流中,块盒子(block-level boxes )和内联盒子(inline boxes)分别从纵向和横向对元素进行布局。块盒子通过垂直的方式一个接一个的排列,盒子之间的距离通过垂直方向的边界(margin-top和margin-bottom)控制(计算距离时要注意盒子外边叠加问题)。内联盒子通过水平方式排列,设置垂自方向的padding、borders和margins无效。水平排列的内联盒子通过行盒子(line box)组织在一起,行盒子的高度总是足够容纳包含在其中的所有盒子,可以通过设置行高(line height)控制行盒子的高度。因此,改变内联盒子尺寸的参数只有水平方向上的borders、padding、margins以及line height。

CSS语法

包含在行盒子中的内联元素

CSS2.1可以设置display属性为inline-block,融合inline和block的属性。在水平方向按内联盒子的方式布局,同时可以像块盒子一样设置widths、heights和垂自方向的 margins、padding。

设置relative的效果

设置relative的效果”position: relative; left: 20px; top: 20px;”

Floats

如何使用浮动模式(floats)


CCS将盒子的float属性设置为leftright,并且将position属性设置为staticrelative

float的非none属性暗示了盒子是block类型,因此,非块类型盒子的display属性会因为设置了float属性而改变为块类型。

当盒子采用floats算法定位时,盒子会从普通流中被抽取出来,向左或向右移动,直到遇到父盒子或设置了float属性盒子的边界。

设置floats的效果

设置float的效果

上图3个红色的方块,两个左浮(float: left),一个右浮。第二个左浮窗口位于第一个左浮动窗口右边,如果再增加左浮窗口,会不断的照此叠加,充满父盒子后会换行继续显示。

一个盒子设置了浮动后,会影响它的兄弟元素,具体的影响方式较为复杂,这要视乎这些兄弟元素是块级元素还是内联元素,若是块级元素会无视这个浮动的块框,使自身尽可能与这个浮动元素处于同一行,导致被浮动元素覆盖,除非这些div设置了宽度,并且父元素的宽度不足以包含它们,这样兄弟元素才会被强制换行;若是内联元素,则会尽可能围绕浮动元素。

浮动元素脱离了普通流,因此包含它的父元素不会因为这个浮动元素的存在而自动撑高,这就造成了高度塌陷。下图所示,由于左边盒子浮动而脱离了普通流,父元素的高度只是由span盒子决定,看起来就像父元素高度塌陷了。

浮动的影响

浮动的影响

当浮动影响到盒子的布局时,需要清除浮动。

Absolute Positioning

如何采用绝对定位(absolute positioning)


CCS将盒子的position属性设置为absolutefixed

当设置为fixed的时候,盒子的位置相对于浏览器可见的视窗固定,即使拖动浏览器的滚动条,盒子的位置也固定不变。

设置absolute的效果

设置absolute的效果”position: absolute; left: 20px; top: 20px;”

清除浮动(clearing floats)

由于浮动元素会影响它的兄弟元素的位置和父元素产生高度塌陷,需要清除浮动。

常用的清除浮动方法是clear: bothclear的属性值bothleftrightnoneinherit 分别代表在元素左右两侧不允许出现浮动元素、左侧不允许出现浮动元素、右侧不允许出现浮动元素、不清除浮动、继承父元素的值。

但是,clear只是清除了浮动对兄弟元素的影响,而高度塌陷问题还没有解决,需要更高级的清除浮动——闭合浮动。为什么叫闭合浮动?因为浮动的元素脱离了普通流,对于它的父元素,它并没有闭合,这时候就需要闭合浮动了。

闭合浮动的3种方法


(1)空div方法

<div class="box">
    <div class="main left">我设置了左浮动 float: left</div>
    <div style="clear: both;"></div>
    <div class="aside">我是页脚,我的上面添加了一个设置了 clear: both 的空 div</div>
</div>

div方法很方便,但是加入了没有涵义的div,这违背了结构与表现分离的原则,并且后期维护也不方便。

(2)overflow方法

<div class="box" style="overflow: hidden; *zoom: 1;">
    <div class="main left">我设置了左浮动 float: left</div>
    <div class="aside left">我是页脚,但是我也设置了左浮动。</div>
</div>

当元素内包含会超出父元素边界的子元素时,overflow方法可能会覆盖掉有用的子元素,或是产生了多余的滚动条。

(3):after伪元素的方法

<style>
    .clearfix {/* 触发 hasLayout */ zoom: 1; }
    .clearfix:after {content: '.'; display: block; height: 0; clear: both; visibility: hidden; }
</style>
<div class="box clearfix">
    <div class="main left">我设置了左浮动 float: left</div>
    <div class="aside left">我是页脚,但是我也设置了左浮动。</div>
</div>

这个办法不但完美兼容主流浏览器,并且也很方便,使用重用的类,可以减轻代码编写,另外网页的结构也会更加清晰。

清除浮动的详细介绍可参考《详说清除浮动》示例)。

块格式化环境

块格式化环境(block formatting context)


块格式化环境是CCS渲染Web页面的一块区域,块盒子在该区域内布局。

块格式化环境对(float)定位和清除(clear)浮动至关重要,定位和清除浮动只对同一块格式化环境中的对象有效。float不会影响到其它块格式化环境中的盒子,clear只清除同一块格式化环境中之前的float效果。

简单来说,块格式化环境是一种属性,这种属性会影响着元素的定位以及与其兄弟元素之间的相互作用。

块格式化环境就是一个作用范围,可理解为一个独立的容器,这个容器的里盒子的布局与这个容器外的不相干。

块格式化环境不存在嵌套包含关系,块格式化环境只包含该环境内的对象,不会再包含该环境中对象再创建的块格式化环境。(A block formatting context contains everything inside of the element creating it that is not also inside a descendant element that creates a new block formatting context.)

创建块格式化环境的条件(满足任意一条即可)


  • the root element or something that contains it;
  • float设置为none
  • position设置为absolutefixed
  • display设置为inline-blocktable-celltable-captionflexinline-flex
  • overflow设置为visible

块格式化环境主要有三个特性(详见《详说 Block Formatting Contexts (块级格式化上下文)》示例):

  1. 阻止外边距折叠;
  2. 包含浮动的元素(清除浮动之overflow方法);
  3. 阻止元素被浮动元素覆盖。

高级选择器

CSS 2.1 属性选择器

E[att] 匹配所有具有att属性的E元素,不考虑它的值。(注意:E在此处可以省略,比如”[cheacked]”。以下同。)

E[att=val] 匹配所有att属性等于”val”的E元素。

E[att~=val] 匹配所有att属性具有多个空格分隔的值、其中一个值等于”val”的E元素。

E[att|=val] 匹配所有att属性具有多个连字号分隔(hyphen-separated)的值、其中一个值以”val”开头的E元素,主要用于lang属性,比如”en”、”en-us”、”en-gb”等等。

示例:

p[title] { color:#f00; }
div[class=error] { color:#f00; }
td[headers~=col1] { color:#f00; }
p[lang|=en] { color:#f00; }
blockquote[class=quote][cite] { color:#f00; }

CSS 2.1 伪类(pseudo-classes)

E:first-child 匹配父元素的第一个子元素。

E:link 匹配所有未被点击的链接。

E:visited 匹配所有已被点击的链接。

E:active 匹配鼠标已经其上按下、还没有释放的E元素。

E:hover 匹配鼠标悬停其上的E元素。

E:focus 匹配获得当前焦点的E元素。

E:lang(c) 匹配lang属性等于c的E元素。

示例:

p:first-child { font-style:italic; }
input[type=text]:focus { color:#000; background:#ffe; }
input[type=text]:focus:hover { background:#fff; }
q:lang(sv) { quotes: "\201D" "\201D" "\2019" "\2019"; }

CSS 2.1 伪元素(pseudo-elements)

E:first-line 匹配E元素的第一行。

E:first-letter 匹配E元素的第一个字母。

E:before 在E元素之前插入生成的内容。

E:after 在E元素之后插入生成的内容。

示例:

p:first-line { font-weight:bold; color;#600; }
.preamble:first-letter { font-size:1.5em; font-weight:bold; }
.cbb:before { content:""; display:block; height:17px; width:18px; background:url(top.png) no-repeat 0 0; margin:0 0 0 -18px; }
a:link:after { content: " (" attr(href) ") "; }

更多关于CSS选选择器类容可参考CSS reference

使用技巧

!important规则


多条CSS语句互相冲突时,具有!important的语句将覆盖其他语句。由于IE不支持!important,所以也可以利用它区分不同的浏览器。

h1 {color: red !important; color: blue;}

容器水平居中


先为该容器设置一个明确宽度,然后将margin的水平值设为auto即可。

div#container {width: 760px; margin: 0 auto;}

禁止自动换行


文字在一行中显示完成,不要自动换行。

h1 { white-space: nowrap; }

图片宽度自适应


如何使得较大的图片,能够自动适应小容器的宽度?

img {max-width: 100%}

设置link状态的顺序


link的四种状态,需要按照下面的前后顺序进行设置。

a:link 
a:visited 
a:hover 
a:active

Text-transform和Font Variant


Text-transform用于将所有字母变成小写字母、大写字母或首字母大写。

p {text-transform: uppercase} 
p {text-transform: lowercase} 
p {text-transform: capitalize}

Font Variant用于将字体变成小型的大写字母(即与小写字母等高的大写字母)。

p {font-variant: small-caps}

用图片充当列表标志


默认情况下,浏览器使用一个黑圆圈作为列表标志,可以用图片取代它。

ul {list-style: none}
ul li {
    background-image: url("path-to-your-image");
    background-repeat: none;
    background-position: 0 0.5em; 
}

用图片替换文字


在标题栏中使用图片,但是又必须保证搜索引擎能够读到标题。

h1 { 
    text-indent: -9999px; 
    background: url("h1-image.jpg") no-repeat; 
    width: 200px;
    height: 50px;
}

预处理器

CSS预处理器(css preprocessor)的基本思想是,用一种专门的编程语言,进行网页样式设计,然后再编译成CSS文件。常用的CSS预处理器有LessSass等。

Less

Less使用变量(variables)、混合(mixins)、函数(functions)和许多其他的技术,让你的CSS更具维护性、主题性、扩展性。Less可运行在Node环境,浏览器环境和Rhino环境,同时也有3种可选工具供你编译文件和监视任何改变。

具体用法可参考Bootstrap中文网的Less教程

Sass & Compass

Sass使用变量(variables)、混合(mixins)、嵌套规则(nested rules)、内联导入(inline imports)等,让CSS更加优雅和强大。Sass让CSS更好的组织、体积更小、速度更快。

Compass是Sass的工具库(toolkit)。Sass本身只是一个编译器,Compass在它的基础上,封装了一系列有用的模块和模板,补充Sass的功能。它们之间的关系,有点像Javascript和jQuery、Ruby和Rails、python和Django的关系。

具体使用可参考阮一峰的SassCompass用法指南。

参考资料

  1. MDN: CSS
  2. W3: Cascading Style Sheets home page
  3. w3school: CSS 教程
  4. W3: Specified, computed, and actual values
  5. W3: CSS Values and Units Module Level 3
  6. W3: CSS Color Module Level 3
  7. W3: CSS basic box model
  8. W3: CSS Grid Layout Module Level 1
  9. W3: CSS Flexible Box Layout Module Level 1
  10. 阮一峰:CSS使用技巧
  11. 阮一峰:CSS选择器笔记
  12. cuishengli:CSS 框模型概述
  13. leejersey:详说 Block Formatting Contexts (块级格式化上下文)
  14. Kayo:详说清除浮动
  15. Jean: 8 CSS preprocessors to speed up development time
  16. LESS « 一种动态样式语言
  17. 阮一峰:SASS用法指南
  18. 阮一峰:Compass用法指南

脚注

  1. 也可参考”CSS Mastery: Advanced Web Standards Solutions, Second Edition”一书的”Chapter 3: Visual Formatting Model Overview”。 ↩
(完)
comments powered by Disqus
Powered by GitHub  &&  Jekyll | CC BY-NC-SA