工作上,除了Django和一些并不复杂的脚本以外,其余时间写了大量的CSS和jQuery,现在回头看之前的一些代码,写得十分丑陋又低效,其中的CSS就有很大改进空间。
正如多数人的认知一样,HTML和CSS并不难学难用,从学习曲线上来说确实如此,难度甚至不如使用VIM。但是写不写得好又是另一回事,好的CSS代码能用最少的代码量实现功能,易修改且性能佳。易修改,举个最简单的例子,比如要求修改一个div的高宽且保持其子div自适应高宽,若是写死了子div的高宽,修改工作很麻烦,所以最好是将子div在需求下尽可能写成自适应,这样修改时就只需要修改父div的高宽即可。性能佳,能用CSS实现的绝对不用js实现,不管是网页布局还是动画效果,原生的CSS都是快速又具备高度兼容性的选择。
清除浮动
清除浮动是个常见问题,不少人的解决办法是添加一个空的 div 应用 clear:both
。事实上仅需要使用after伪类即可在元素尾部自动清除浮动
.clear-fix { overflow: hidden; zoom: 1; }
.clear-fix:after { display: table; content: ""; width: 0; clear: both; }
DIV同行排列
最容易想到的是将一行div全设置为display:inline-block
,但这种做法会使得两个div之间存在“间隔”,这个“间隔”的大小通常由font-size决定,将font-size设为0将不存在"间隔"。清除间隔可以通过使用注释的方法实现。
<div class="item"></div><!--
--><div class="item"></div>
更好的方式自然还是使用float
.item {float: left}
灵活使用BFC
BFC(Block Formatting Context)直译为“块级格式化范围”。当一个HTML元素满足以下任何一点时,就会产生BFC:
float
的值不为none
overflow
的值不为visible
display
的值为table-cell
,table-caption
或inline-block
position
的值不为relative
和static
BFC提供了一个环境,这个环境中的元素不会影响到其它环境中的布局。比如浮动元素形成BFC,浮动元素内部子元素的主要受该浮动元素影响,两个浮动元素之间是互不影响的。BFC就是一个作用范围,可看作是一个独立的容器,并且这个容器的布局,与这个容器外的元素毫不相干。
BFC的元素不能与浮动元素重叠,当容器有足够的剩余空间容纳 BFC 的宽度时,所有浏览器都会将 BFC 放置在浮动元素所在行的剩余空间内。
上图所示,画线左侧是描述,字数不定,即宽度不定,画线占满同一行的剩余部分。正好使用上面提到的BFC与浮动元素的位置情况。
.desc { float: left;}
.line { overflow: hidden;}
<div>
<div class="desc">黄金档</div>
<div class="line"></div>
</div>
未垂直对齐
同一行的一组class为item的div使用了display:inline-block
或者是float:left
时,如果某个div的内部标签中填充一些文字等内容,可能就会出现垂直不对齐的情况。但非常奇怪的是,这时内部元素并没有超出父级div的范围,没有任何溢出或撑开的情况,这点我也不是很理解,知道其发生原因的同学欢迎留言。解决方法倒不难:
.item { vertical-align: top;}
display:table-cell的应用
table-cell会被其他一些CSS属性破坏,例如float
和 position:absolute
,所以display:table-cell
与float:left
或是position:absolute
属性最好不要同用。设置了该属性的元素对宽度高度敏感,响应padding属性,对margin值无反应
垂直居中
.content {
display: table-cell;
border: 1px solid #eee;
width: 600px;
text-align: center;
}
<div class="content">
<p>what a beautiful day</p>
</div>
两栏自适应布局
适用于一栏宽度不固定,比如大小不确定的图片,另一栏自动调整占满剩余宽度的场景。
.box {
width: 70%;
}
.content {
display: table-cell;
border: 1px solid #eee;
}
.fix {
float: left;
color: #a8c;
}
<div class="box">
<div class="fix">This is left fixed block</div>
<div class="content">
风住尘香花已尽,日晚倦梳头。物是人非事事休,欲语泪先流。闻说双溪春尚好,也拟泛轻舟。只恐双溪舴艋舟,载不动许多愁。
</div>
</div>
垂直对齐的trick
上面提到过可以用table-cell让div内的元素垂直居中,但对div的float
和position
属性有限制,现在介绍一种可以用在浮动元素(容器)里的垂直居中方法,这是在工作中碰到使用了float不好使时发现的
使用一个用作协助的inline-block
的帮助元素,此处是span并将该元素和准备居中的元素都设置为vertical-align: middle
即可
<div class="frame">
<span class="helper">
<img src="http://jsfiddle.net/img/logo.png" />
</div>
css
.frame {
float: left;
height: 220px;
width: 300px;
}
.helper {
display: inline-block;
height: 100%;
vertical-align: middle;
}
img {
vertical-align: middle;
max-height: 25px;
max-width: 160px;
}
原理是相邻的两个inline-block
元素可以互相对齐,所以使用一个不可见的100%高度的span可以帮助img对齐
来源:http://stackoverflow.com/questions/7273338/how-to-vertically-align-an-image-inside-div
不过这时仍然有个问题,假设上面的img后面要跟一个文字介绍的div块(inline-block
),如下所示
<div class="frame">
<span class="helper">
<img src="http://jsfiddle.net/img/logo.png" />
<div class="desc">ABCD</div>
</div>
如果需求是设置.desc
的margin-top: 10px
,直接设置的话会发现前面居中对齐的img
也会向下10px。
原因是因为块元素的vertical-align
默认是baseline
,设置margin-top
会导致整个baseline
下降,所以之前已经居中对齐的元素也会受到影响。解决办法是将div.desc
设置为其它的对齐方式,推荐vertical-align: top
另外一种解决办法既能解决上面的问题,也能解决inline-block
的块元素设置margin
无效的问题
.desc {
postion: relative;
top: 10px;
}
hover时显示边框
方法一
对于一个块元素比如div,通常先设置好border-width
,并将border-color
设为#fff
或transparent
,hover时直接改变border的颜色即可
方法二
将宽高的值写上,然后设置box-sizing: box
,hover时直接设置border
即可,此时边框宽度包括在width
或height
的数值中
**粗体** _斜体_ [链接](http://example.com) `代码` - 列表 > 引用
。你还可以使用@
来通知其他用户。