第 4 章
其他基础知识与Ionic项目结构
本书第3章介绍的AngularJS技术是一个结构复杂庞大、组件配合紧密的框架,熟练掌握了AngularJS就已经基本上跨过了Ionic开发的门槛。不过除了提供AngularJS功能组件外,Ionic还为前端组件定制了美观的样式,并使用业内流行的前端工具整合了自动化的项目开发工具链。
因此在全面介绍Ionic的组件和开发前,安排了本章介绍掌握Ionic开发需要了解的SASS样式开发和构建工具Gulp。*后将Ionic项目的整体目录文件结构做一个说明,这样读者未来在需要开发或是阅读调试代码时,就知道该到什么位置去查看了,而不是漫无目的地凭直觉瞎找。
本章的主要知识点包括:
? SASS基础知识
? lodash库简单说明
? Gulp原理与常用模块介绍
? Ionic项目模板目录结构解析
4.1 SASS 入门
SASS是一种对CSS进行了扩充的开发工具,它提供了许多便利的写法,使得CSS的开发变得简单和可维护,大大节省了样式设计者尤其是有编程背景的样式设计者的时间。符合SASS语法的文件就是普通的文本文件,里面可以直接使用CSS语法。SASS文件后缀名是.scss,意思为Sassy CSS。因此有时候SASS和SCSS两个词是可以混用的。
Ionic提供的样式文件就是基于SASS开发的。考虑到部分读者从未接触过SASS,本书将重点介绍Ionic涉及的SASS语法,并不打算变成一个完整的SASS说明文档。有通读需要的读者可以到SASS的官方网站学习SASS的更多特性和样例:http://sass-lang.com/documentation/file.SASS_REFERENCE.html。
编写完成的SASS文件需要经过编译处理转换成浏览器可以识别的CSS代码,在Ionic里有本章4.3节介绍的Gulp调用相关模块完成编译。在开发者日常编写调试时,可以使用一个在线SASS服务网站(http://www.sassmeister.com/)的即时编译转换功能获得CSS代码,如图4.1所示。
图4.1 使用在线网站(http://www.sassmeister.com/)的即时编译转换功能获得CSS代码
4.1.1 变量与计算
SASS允许定义变量,变量需要冠以$前缀,如:
$period : 1s;
$effect : ease-in;
$trans_property : all;
a {
-moz-transition: $trans_property $period $effect;
-webkit-transition: $trans_property $period $effect;
-o-transition: $trans_property $period $effect;
transition: $trans_property $period $effect;
}
经转换后的CSS代码为:
a {
-moz-transition: all 1s ease-in;
-webkit-transition: all 1s ease-in;
-o-transition: all 1s ease-in;
transition: all 1s ease-in;
}
【代码解析】从代码上看似乎使用SASS变量的源代码更长,但是有了变量遇到以后的调整变化时,就只需要在变量定义的地方变更值,而不用通过全文搜索去替换。相信有过网站维护经验的读者能够体会SASS变量的好处。这也是Ionic在定义CSS样式类使用的*常见模式。
如果变量需要镶嵌在字符串之中,就必须需要写在#{}之中,如:
$side : left;
$default_radius : 5px;
.rounded {
border-#{$side}-radius: $default_radius;
}
经转换后的CSS代码为:
.rounded {
border-left-radius: 5px;
}
【代码解析】这种字符串替换经常被使用在组合型的CSS属性名上。
SASS允许在代码中使用计算表达式,如:
$var : 2;
$more_px : 10px;
body {
margin: (16px/2);
top: 100px + 5 * $more_px;
right: $var * 10%;
}
经转换后的CSS代码为:
body {
margin: 8px;
top: 150px;
right: 20%;
}
【代码解析】变量也可以出现在计算表达式中,这样就更灵活了。
4.1.2 样式嵌套
标准的CSS只能支持单层的选择器{}块结构,对于习惯了JavaScript开发的人来说无疑是值得改进的一个地方。而经SASS扩展,可以允许无限层的选择器嵌套,如:
$default_font_size: 100%;
.container {
h1 {
color:red;
font-size: $default_font_size * 2;
}
h2 {
color:blue;
font-size: $default_font_size * 1.5;
}
}
经转换后的CSS代码为:
.container h1 {
color: red;
font-size: 200%;
}
.container h2 {
color: blue;
font-size: 150%;
}
【代码解析】从代码可以看到,生成后的CSS代码是松散的平面结构,而SASS的代码明显更有逻辑性。
CSS属性名也可以嵌套生成,如:
div.container {
border: {
color: green;
}
border-left: {
color: red;
}
}
经转换后的CSS代码为:
div.container {
border-color: green;
border-left-color: red;
}
【代码解析】从代码可以看到,在border和border-left后分别加上冒号后,生成的CSS会使用-号来连接生成*终的属性名。
在嵌套的代码块内,可以使用&占位符表示引用父元素。如:
a {
&:link { color: blue; }
&:visited { color: green; }
&:active { color: blue; }
&:hover {
color: red;
font-weight: bold;
}
}
经转换后的CSS代码为:
a:link {
color: blue;
}
a:visited {
color: green;
}
a:active {
color: blue;
}
a:hover {
color: red;
font-weight: bold;
}
【代码解析】从本示例代码的里可以看出使用SASS的深层嵌套在属性较多时有可能可以减少编写的代码量,代码结构也更具有可读性。
4.1.3 单行注释 //
SASS是CSS的超集,因此标准的CSS注释 /* comment */ ,会保留到编译后生成的文件。而为了方便开发人员的调试,SASS支持了类似JavaScript的单行注释符//,如:
/*
这是单行注释,将被保留
*/
p{
color: red; // 单行注释示例
font-size: 10px; /* CSS原生注释风格示例 */
}
经转换后的CSS代码为:
/*
这是单行注释,将被保留
*/
p {
color: red;
font-size: 10px;
/* CSS原生注释风格示例 */
}
【代码解析】*终在生成的CSS代码里,标准的CSS注释被保留,单行注释符//被忽略省去,出于保护目的不愿把内部注释发布到网上的开发者也可以考虑使用这个方法。
4.1.4 继承@extend
SASS允许一个选择器继承另一个选择器,如:
.classParent1{
border: 1px solid #ddd;
}
.classParent2{
color: red;
text-align: center;
}
.classChild {
@extend .classParent1;
@extend .classParent2;
font-size:120%;
}
p {
@extend .classParent1;
@extend .classParent2;
font-size:120%;
}
经转换后的CSS代码为:
.classParent1, .classChild, p {
border: 1px solid #ddd;
}
.classParent2, .classChild, p {
color: red;
text-align: center;
}
.classChild {
font-size: 120%;
}
p {
font-size: 120%;
}
【代码解析】这里可以看到SASS跟CSS代码相比的好处是既通过@extend继承了父CSS类的样式属性,又把相关的声明都放在子CSS类或子元素声明里,这样的代码结构可阅读可维护性明显更佳。
此处的通过@extend只能继承CSS类,即父类只能是CSS类,而不能是元素。
4.1.5 混入@mixin与@include
*早的SASS是用Ruby开发的,因此该语言的作者引入了一些类似Ruby的语言结构,其中就有用于实现多重继承的混入(Mixin)。混入有点像C语言的宏,是可以定义以后在被引入的地方展开而达到重用的代码块。
首先需要使用@mixin命令,定义一个代码块,随后再使用@include命令,调用这个混入代码块使之原地展开,如:
$border-width : 1px;
@mixin left-setting {
float: left;
margin-left: 10px;
padding-left: 2px;
border-left: $border-width;
}
div {
@include left-setting;
}
经转换后的CSS代码为:
div {
float: left;
margin-left: 10px;
padding-left: 2px;
border-left: 1px;
}
【代码解析】如代码所示,混入定义本身并不生成CSS代码,它类似于静态库被嵌入,当一个元素或者CSS类引入了多个混入代码块,则就相当于实现了多重继承的概念了。
此处变量$border-width的定义位置需要在名为left-setting的混入之前,否则将无法获取该变量的值。这种要求是SASS编译器本身的限制导致的。
混入还可以指定参数和默认值,既像C语言的宏又强于它,如:
@mixin left-setting($border-width: 3px) {
float: left;
margin-left: 10px;
padding-left: 2px;
border-left: $border-width;
}
div {
@include left-setting;
}
div.special{
@include left-setting(5px);
}
经转换后的CSS代码为:
div {
float: left;
margin-left: 10px;
padding-left: 2px;
border-left: 3px;
……