AngularJs自定义指令详解(6) - controller、require
在前面文章中提到一旦声明了require,则链接函数具有第四个参数:controller。
可见require和controller是配合使用的。
在自定义指令中使用controller,目的往往是要封装一些行为,给其他指令使用。下面是一个简单的例子:
<!DOCTYPE html>
<html>
<head lang="en">
<meta charset="UTF-8">
<script src="../lib/angular-1.3.16/angular.min.js"></script>
<title></title>
<script language="JavaScript">
var app = angular.module('myapp',[]); app.directive('d1',function() {
return {
controller: function ($scope) {
this.method1 = function () {
return 'World';
};
}
}
}); app.directive('d2',function() {
return {
require: 'd1',
link: function (scope, elem, attrs, d1) {
scope.greeting = d1.method1();
}
}
});
</script>
</head>
<body ng-app="myapp">
<div d1 d2>Hello {{greeting}}!</div>
</body>
</html>
输出:
Hello World!
从这个例子可以看到,在DOM中需要声明d1、d2,如果不声明d1,那么d2在检查require里的'd1'时,就会抛出异常。
AngularJs为我们提供了一些标记,使用它们可以告诉AngularJs怎么查找所require的控制器:
1,没标记。在当前元素中查找,如果找不到就抛出错误。
2,?标记。在当前元素中查找,如果查找不到,不抛出错误,链接函数的第四个参数为null(如上面代码link: function (scope, elem, attrs, d1)中的d1)
3,^标记。不仅在当前元素中查找,还查找其所有父级。如果找不到就抛出错误。
4,^^标记。只在当前元素的父级中查找。如果找不到就抛出错误。
5,?^标记。不仅在当前元素中查找,还查找其所有父级。如果查找不到,不抛出错误,链接函数的第四个参数为null
6,?^^标记。只在当前元素的父级中查找。如果查找不到,不抛出错误,链接函数的第四个参数为null
我们修改一下上面例子的代码,试验一下第4条:
<!DOCTYPE html>
<html>
<head lang="en">
<meta charset="UTF-8">
<script src="../lib/angular-1.3.16/angular.min.js"></script>
<title></title>
<script language="JavaScript">
var app = angular.module('myapp',[]); app.directive('d1',function() {
return {
controller: function ($scope) {
this.method1 = function () {
return 'World';
};
}
}
}); app.directive('d2',function() {
return {
require: '^^d1',
link: function (scope, elem, attrs, d1) {
scope.greeting = d1.method1();
}
}
});
</script>
</head>
<body ng-app="myapp">
<div d1 d2>Hello {{greeting}}!</div>
</body>
</html>
在Chrome浏览器的控制台可以看到错误提示:
Error: [$compile:ctreq] http://errors.angularjs.org/1.3.16/$compile/ctreq?p0=d1&p1=d2
点击链接去看看(需要翻墙):
Error: $compile:ctreq
Controller 'd1', required by directive 'd2', can't be found!
果然找不到!因为d1放在d2所在的div中(当前元素),^^标记要求去父级找,当然没有了。
在实际应用时,我们往往会require 'ngModel',也就是希望利用AngularJs内置指令ngModel里的方法,而不是自己重新写:
<!DOCTYPE html>
<html>
<head lang="en">
<meta charset="UTF-8">
<script src="../lib/angular-1.3.16/angular.min.js"></script>
<title></title>
<script language="JavaScript">
var app = angular.module('myApp',[]); app.directive('myDirective',function() {
return {
require: 'ngModel',
link: function (scope, elem, attrs, model) {
model.$parsers.unshift(function(value) {
if (parseFloat(value)<1) {
model.$setValidity('test', true);
return parseFloat(value).toFixed(2);
} else {
model.$setValidity('test', false);
return undefined;
}
});
}
}
}); </script>
</head>
<body ng-app="myApp">
<form name="form1">
<div> 请输入小于1的一个小数:<input name="input1" type="number" ng-model="aNumber" my-directive />
保留小数点后2位:{{aNumber}}
</div>
<span ng-show="form1.input1.$error.test">这个数并不小于1!</span>
</form>
</body>
</html>
上面例子就是利用了ngModel里已有的$parser属性和$setValidity方法。
$parsers里保存了一组function, 每当DOM里数据变化的时候, 这组function会被调用。
AngularJs自定义指令详解(6) - controller、require的更多相关文章
- AngularJs自定义指令详解(1) - restrict
下面所有例子都使用angular-1.3.16.下载地址:http://cdn.bootcss.com/angular.js/1.3.16/angular.min.js 既然AngularJs快要发布 ...
- AngularJs自定义指令详解(2) - template
一些用于定义行为的指令,可能不需要使用template参数. 当指定template参数时,其值可以是一个字符串,表示一段HTML文本,也可以是一个函数,这函数接受两个参数:tElement和tAtt ...
- AngularJs自定义指令详解(5) - link
在指令中操作DOM,我们需要link参数,这参数要求声明一个函数,称之为链接函数. 写法: link: function(scope, element, attrs) { // 在这里操作DOM} 如 ...
- AngularJs自定义指令详解(8) - priority
priority 默认值为0. 当一个元素上声明两个指令,而且它们的priority一样,谁先被调用?这个需要分情况讲.下面先给个例子: <!DOCTYPE html> <html& ...
- AngularJs自定义指令详解(3) - scope
我们之所以要定义指令,目的是重用指令.假设有这么一个应用场景:在同一个html里使用了两次my-directive,第一个my-directive要展示的是Hello World,第二个my-dire ...
- AngularJs自定义指令详解(10) - 执行次序
代码: <!DOCTYPE html> <html> <head lang="en"> <meta charset="UTF-8 ...
- AngularJs自定义指令详解(9) - terminal
例子: <!DOCTYPE html> <html> <head lang="en"> <meta charset="UTF-8 ...
- AngularJs自定义指令详解(7) - multiElement
multiElement不太常用,从下面这个例子可以大致看出它的作用: <!DOCTYPE html> <html> <head lang="en"& ...
- AngularJs自定义指令详解(4) - transclude
transclude默认值为false,如果设置 transclude为true,那么相应地,必须在模板代码中加入ng-transclude指令. 先看个例子: <!DOCTYPE html&g ...
随机推荐
- Create a Listlink
#ifndef List_h__ #define List_h__ #include <stdio.h> struct ListNode { int value; ListNode* pN ...
- PVANET----Deep but Lightweight Neural Networks for Real-time Object Detection论文记录
arxiv上放出的物体检测的文章,在Pascal voc数据集上排第二.源码也已放出(https://github.com/sanghoon/pva-faster-rcnn),又可以慢慢把玩了.这篇文 ...
- java学习第20天(IO流)
构造方法File file = new File("e:\\demo"); 创建文件夹 File file = new File("e:\\demo"); fi ...
- 关于<textArea>控件下显示不出其它控件
今天在写页面控件时发现我添加一个<textarea>后,在其下方的控件都显示不出来了,后来发现我忘写结束标记</textarea>了 以后该写结束标记还要写啊!
- jdk1.7和jdk1.8的String的getByte方法的差异
最近遇到一个奇葩的bug,jdk1.7下正常的程序到了jdk1.8下就不能用了,经过查找原因发现是因为jdk版本升级导致的获取的getbyte时得到的byte数组不同造成的.
- 动态组合lambda 表达式
//记录实体集合—动态组合lambda 表达式 Expression<Func<AdEntity, bool>> thirdWhere = p => p.Observer ...
- JavaEE第一天知识点总结
JavaEE第一天知识点总结 持久化是将程序中的数据在瞬时状态和持久状态间转换的机制 DAO: 英文全称: Date Access Object(数据存取对象) 位于业务逻辑和持久化数据之间 实现对持 ...
- python的类与对象
类与对象 1.什么是类 类和函数一样是程序编程的一种方式,在处理某些问题的时候类比函数更加适合让编程变得简化,在python里面函数式编程和类编程都是为了简化代码的一种编程方式,具体应用那个则由具体问 ...
- 一步一步学WebSocket(二) 使用SuperWebSocket实现自己的服务端
上一篇文章,我们了解了客户端如何与服务器创建WebSocket连接.但是一个巴掌拍不响,既然是通信,就必然最少要有两个端.今天我们来看看c#如何用已有的框架实现一个WebSocket服务端. 在.Ne ...
- vs2012启动网站调试提示 HTTP 错误 403.14 - Forbidden Web 服务器被配置为不列出此目录的内容
启动vs2012调试网站的时候提示: HTTP 错误 403.14 - Forbidden Web 服务器被配置为不列出此目录的内容. 最可能的原因: 没有为请求的 URL 配置默认文档,并且没有在服 ...