过滤器用来格式化需要展示给用户的数据。AngularJS有很多实用的内置过滤器,同时也提供了方便的途径可以自己创建过滤器。
在HTML中的模板绑定符号{{ }}
内通过 | 符号来调用过滤器。如果要同时使用多个过滤器,可以用 | 符号作为分割符来使用多个过滤器。例如,假设我们希望将字符串转换成大写,可以对字符串中的每个字符都单独进行转换操作,也可以使用过滤器:{{ name | uppercase }}
。
在JavaScript代码中可以通过$filter
来调用过滤器。例如,在JavaScript代码中使用lowercase
过滤器:
app.controller('DemoController', ['$scope', '$filter',
function($scope, $filter) {
$scope.name = $filter('lowercase')('Ari');
}]);
以HTML的形式使用过滤器时,如果需要传递参数给过滤器,只要在过滤器名字后面加冒号即可。如果有多个参数,可以在每个参数后面都加入冒号。例如,数值过滤器可以限制小数点后的位数,在过滤器后写上 :2 可以将2作为参数传给过滤器:
{{ 123.456789 | number:2 }}
内置过滤器
currency
currecy过滤器可以将一个数值格式化为货币格式。
{{ 123456 | currency }}
currecy过滤器允许我们自己设置货币符号。默认情况下会采用客户端所处区域的货币符号,但是也可以自定义货币符号。给currency
传递一个字符串类型的参数,就可以自定义货币符号了。
{{ 123456 | currency:'¥' }}
date
date过滤器可以将日期格式化成需要的格式。AngularJS中内置了几种日期格式,如果没有指定使用任何格式,默认会采用mediumDate
格式。
下面是内置的支持本地化的日期格式:
{{ today | date:'medium' }}
{{ today | date:'short' }}
{{ today | date:'fullDate' }}
{{ today | date:'longDate' }}
{{ today | date:'mediumDate' }}
{{ today | date:'shortDate' }}
{{ today | date:'mediumTime' }}
{{ today | date:'shortTime' }}
// 年份格式化
四位年份:{{ today | date:'yyyy' }}
两位年份:{{ today | date:'yy' }}
一位年份:{{ today | date:'y' }}
// 月份格式化
英文月份:{{ today | date:'MMMM' }}
英文月份简写:{{ today | date:'MMM' }}
数字月份:{{ today |date:'MM' }}
一年中的第几个月份:{{ today |date:'M' }}
// 日期格式化
数字日期:{{ today|date:'dd' }}
一个月中的第几天:{{ today | date:'d' }}
英文星期:{{ today | date:'EEEE' }}
英文星期简写:{{ today | date:'EEE' }}
// 小时格式化
24小时制数字小时:{{today|date:'HH'}}
一天中的第几个小时:{{today|date:'H'}}
12小时制数字小时:{{today|date:'hh'}}
上午或下午的第几个小时:{{today|date:'h'}}
// 分钟格式化
数字分钟数:{{ today | date:'mm' }}
一个小时中的第几分钟:{{ today | date:'m' }}
// 秒数格式化
数字秒数:{{ today | date:'ss' }}
一分钟内的第几秒:{{ today | date:'s' }}
毫秒数:{{ today | date:'.sss' } }
// 字符格式化
上下午标识:{{ today | date:'a' }}
四位时区标识:{{ today | date:'Z' }}
// 一些自定义日期格式:
{{ today | date:'MMMd, y' }}
{{ today | date:'EEEE, d, M' }}
{{ today | date:'hh:mm:ss.sss' }}
filter
filter过滤器可以从给定数组中选择一个子集,并将其生成一个新数组返回。这个过滤器通常用来过滤需要进行展示的元素。
这个过滤器的第一个参数可以是字符串、对象或是一个用来从数组中选择元素的函数。下面分情况介绍传入不同类型的参数。
- 字符串
返回所有包含这个字符串的元素。如果我们想返回不包含该字符串的元素,在参数前加 ! 符号。 - 对象
AngularJS会将待过滤对象的属性同这个对象中的同名属性进行比较,如果属性值是字符串就会判断是否包含该字符串。如果我们希望对全部属性都进行对比,可以将 $ 当作键名。 - 函数
对每个元素都执行这个函数,返回非假值的元素会出现在新的数组中并返回。
例如,用下面的过滤器可以选择所有包含字母e的单词:
{{ ['Ari','Lerner','Likes','To','Eat','Pizza'] | filter:'e' }}
如果要过滤对象,可以使用对象过滤器。例如,如果有一个由people对象组成的数组,每个对象都含有他们最喜欢吃的食物的列表,那么可以用下面的形式进行过滤:
{{ [{
'name': 'Ari',
'City': 'San Francisco',
'favorite food': 'Pizza'
},{
'name': 'Nate',
'City': 'San Francisco',
'favorite food': 'indian food'
}] | filter:{'favorite food': 'Pizza'} }}
也可以用自定义函数进行过滤:
{{ ['Ari','likes','to','travel'] | filter:isCapitalized }}
isCapitalized
函数的功能是根据首字母是否为大写返回 true 或 false,具体如下所示:
$scope.isCapitalized = function(str) {
return str[0] == str[0].toUpperCase();
};
我们也可以给filter过滤器传入第二个参数,用来指定预期值同实际值进行比较的方式。
第二个参数可以是以下三种情况之一。
- true
用angular.equals(expected, actual)
对两个值进行严格比较。 - false
进行区分大小写的子字符串比较。 - 函数
运行这个函数,如果返回真值就接受这个元素。
json
json过滤器可以将一个JSON或JavaScript对象转换成字符串。这种转换对调试非常有帮助:
{{ {'name': 'Ari', 'City': 'SanFrancisco'} | json }}
limitTo
limitTo
过滤器会根据传入的参数生成一个新的数组或字符串,新的数组或字符串的长度取决于传入的参数,通过传入参数的正负值来控制从前面还是从后面开始截取。如果传入的长度值大于被操作数组或字符串的长度,那么整个数组或字符串都会被返回。
例如,我们可以截取字符串的前三个字符:
{{ 'San Francisco is very cloudy' | limitTo:3 }}
或最后的六个字符:
{{ 'San Francisco is very cloudy' | limitTo:-6 }}
对数组也可以进行同样的操作。返回数组的第一个元素:
{{ ['a','b','c','d','e','f'] | limitTo:1 }}
lowercase
lowercase过滤器将字符串转为小写。
{{ "San Francisco is very cloudy" | lowercase }}
number
number过滤器将数字格式化成文本。它的第二个参数是可选的,用来控制小数点后截取的位数。如果传入了一个非数字字符,会返会空字符串。
{{ 123456789 | number }}
{{ 1.234567 | number:2 }}
orderBy
orderBy过滤器可以用表达式对指定的数组进行排序。
orderBy可以接受两个参数,第一个是必需的,第二个是可选的。
第一个参数是用来确定数组排序方向的谓词。
下面分情况讨论第一个参数的类型。
- 函数
当第一个参数是函数时,该函数会被当作待排序对象的getter
方法。 - 字符串
对这个字符串进行解析的结果将决定数组元素的排序方向。我们可以传入 + 或 - 来强制进行升序或降序排列。 - 数组
在排序表达式中使用数组元素作为谓词。对于与表达式结果并不严格相等的每个元素,则使用第一个谓词。
第二个参数用来控制排序的方向(是否逆向)。
例如,我们将下面的对象数组用name字段进行排序:
{{ [{
'name': 'Ari',
'status': 'awake'
},{
'name': 'Q',
'status': 'sleeping'
},{
'name': 'Nate',
'status': 'awake'
}] | orderBy:'name' }}
也可以对排序结果进行反转。例如,通过将第二个参数设置为true可以将排序结果进行反转:
{{ [{
'name': 'Ari',
'status': 'awake'
},{
'name': 'Q',
'status': 'sleeping'
},{
'name': 'Nate',
'status': 'awake'
}] | orderBy:'name':true }}
uppercase
uppercase过滤器可以将字符串转换为大写形式:
{{ "San Francisco is very cloudy" | uppercase }}
自定义过滤器
创建自定义过滤器需要将它放到自己的模块中。下面我们来实现一个过滤器,将字符串第一个字母转换为大写。
angular.module('myApp.filters', [])
.filter('capitalize', function() {
return function(input) {
// input是我们传入的字符串
if (input) {
return input[0].toUpperCase() + input.slice(1);
}
});
过滤器本质上是一个会把我们输入的内容当作参数传入进去的函数。
现在,如果想将一个句子的首字母转换成大写形式,可以用过滤器先将整个句子都转换成小写,再把首字母转换成大写:
{{ 'ginger loves dog treats' | lowercase | capitalize }}
表单验证
要使用表单验证,首先要确保表单有一个name属性。所有输入字段都可以进行基本的验证,比如最大、最小长度等。这些功能是由新的HTML5表单属性提供的。
如果想要屏蔽浏览器对表单的默认验证行为,可以在表单元素上添加novalidate
标记。
下面看一下可以在input元素上使用的所有验证选项。
1.必填项
验证某个表单输入是否已填写,只要在输入字段元素上添加HTML5标记required
即可:
2.最小长度
验证表单输入的文本长度是否大于某个最小值,在输入字段上使用AngularJS指令ng-minlength="{number}"
:
3. 最大长度
验证表单输入的文本长度是否小于或等于某个最大值,在输入字段上使用AngularJS指令ng-maxlength="{number}"
:
4. 模式匹配
使用ng-pattern="/PATTERN/"
来确保输入能够匹配指定的正则表达式:
5. 电子邮件
验证输入内容是否是电子邮件:
6. 数字
验证输入内容是否是数字:
7. URL
验证输入内容是否是URL:
8. 在表单中控制变量
表单的属性可以在其所属的$scope对象中访问到,而我们又可以访问$scope对象,因此JavaScript可以间接地访问DOM中的表单属性。借助这些属性,我们可以对表单做出实时响应。这些属性包括下面这些。
(注意,可以使用下面的格式访问这些属性。)
formName.inputFieldName.property
-
未修改的表单
这是一个布尔属性,用来判断用户是否修改了表单。如果未修改,值为true,如果修改过值为false:
formName.inputFieldName.$pristine
-
修改过的表单
只要用户修改过表单,无论输入是否通过验证,该值都返回true:
formName.inputFieldName.$dirty
-
合法的表单
这个布尔型的属性用来判断表单的内容是否合法。如果当前表单内容是合法的,下面属性的值就是true:
formName.inputFieldName.$valid
-
不合法的表单
这个布尔属性用来判断表单的内容是否不合法。如果当前表单内容是不合法的,下面属性的值为true:
formName.inputFieldName.$invalid
-
错误
这是AngularJS提供的另外一个非常有用的属性:$error
对象。它包含当前表单的所有验证内容,以及它们是否合法的信息。用下面的语法访问这个属性:
formName.inputfieldName.$error
如果验证失败,这个属性的值为true;如果值为false,说明输入字段的值通过了验证。
9. 一些有用的CSS样式
AngularJS处理表单时,会根据表单当前的状态添加一些CSS类。
它们包括:
.ng-pristine {}
.ng-dirty {}
.ng-valid {}
.ng-invalid {}
它们对应着表单输入字段的特定状态。
当某个字段中的输入非法时,.ng-invlid
类会被添加到这个字段上:
input.ng-invalid {
border: 1px solid red;
}
input.ng-valid {
border: 1px solid green;
}
-
$parsers
当用户同控制器进行交互,并且ngModelController
中的$setViewValue()
方法被调用时,$parsers
数组中的函数会以流水线的形式被逐个调用。第一个$parse
被调用后,执行结果会传递给第二个$parse
,以此类推。
这些函数可以对输入值进行转换,或者通过$setValidity()
函数设置表单的合法性。
使用$parsers
数组是实现自定义验证的途径之一。例如,假设我们想要确保输入值在某两个数值之间,可以在$parsers
数组中入栈一个新的函数,这个函数会在验证链中被调用。每个$parser
返回的值都会被传入下一个$parser
中。当不希望数据模型发生更新时返回undefined
。
angular.module('myApp')
.directive('oneToTen', function() {
return {
require: '?ngModel',
link: function(scope, ele, attrs, ngModel) {
if (!ngModel) return;
ngModel.$parsers.unshift(
function(viewValue) {
var i = parseInt(viewValue);
if (i >= 0 && i < 10) {
ngModel.$setValidity('oneToTen', true);
return viewValue;
} else {
ngModel.$setValidity('oneToTen', false);
return undefined;
}
});
}
};
});
-
$formatters
当绑定的ngModel
值发生了变化,并经过$parsers
数组中解析器的处理后,这个值会被传递给$formatters
流水线。同$parsers
数组可以修改表单的合法性状态类似,$formatters
中的函数也可以修改并格式化这些值。
比起单纯的验证目的,这些函数更常用来处理视图中的可视变化。例如,假设我们要对某个值进行格式化。通过$formatters
数组可以在这个值上执行过滤器:
angular.module('myApp')
.directive('oneToTen', function() {
return {
require: '?ngModel',
link: function(scope, ele, attrs, ngModel) {
if (!ngModel) return;
ngModel.$formatters.unshift(function(v) {
return $filter('number')(v);
});
}
};
});
10. 组合实例
下面我们一起创建一个注册表单。表单中包括用户的名字、邮件地址以及用户名。
下面开始定义表单:
这个表单的名称是signup_form
,当表单提交时我们要调用signupForm()
。
下面添加用户的名字:
我们添加了一个表单,这个表单有一个名为name的输入字段,并且这个输入字段被ng-model
指令绑定到了$scope
对象的signup.name
上。
用$dirty
属性来确保用户未对输入内容进行修改时错误内容不会显示出来:
Your name is required.
Your name is required to be at least 3 characters
Your name cannot be longer than 20 characters