在前面文章中提到一旦声明了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

Missing Required Controller
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的更多相关文章

  1. AngularJs自定义指令详解(1) - restrict

    下面所有例子都使用angular-1.3.16.下载地址:http://cdn.bootcss.com/angular.js/1.3.16/angular.min.js 既然AngularJs快要发布 ...

  2. AngularJs自定义指令详解(2) - template

    一些用于定义行为的指令,可能不需要使用template参数. 当指定template参数时,其值可以是一个字符串,表示一段HTML文本,也可以是一个函数,这函数接受两个参数:tElement和tAtt ...

  3. AngularJs自定义指令详解(5) - link

    在指令中操作DOM,我们需要link参数,这参数要求声明一个函数,称之为链接函数. 写法: link: function(scope, element, attrs) { // 在这里操作DOM} 如 ...

  4. AngularJs自定义指令详解(8) - priority

    priority 默认值为0. 当一个元素上声明两个指令,而且它们的priority一样,谁先被调用?这个需要分情况讲.下面先给个例子: <!DOCTYPE html> <html& ...

  5. AngularJs自定义指令详解(3) - scope

    我们之所以要定义指令,目的是重用指令.假设有这么一个应用场景:在同一个html里使用了两次my-directive,第一个my-directive要展示的是Hello World,第二个my-dire ...

  6. AngularJs自定义指令详解(10) - 执行次序

    代码: <!DOCTYPE html> <html> <head lang="en"> <meta charset="UTF-8 ...

  7. AngularJs自定义指令详解(9) - terminal

    例子: <!DOCTYPE html> <html> <head lang="en"> <meta charset="UTF-8 ...

  8. AngularJs自定义指令详解(7) - multiElement

    multiElement不太常用,从下面这个例子可以大致看出它的作用: <!DOCTYPE html> <html> <head lang="en"& ...

  9. AngularJs自定义指令详解(4) - transclude

    transclude默认值为false,如果设置 transclude为true,那么相应地,必须在模板代码中加入ng-transclude指令. 先看个例子: <!DOCTYPE html&g ...

随机推荐

  1. OC编程的一些UI细节

    1/如果你想用一个半透明的View遮住当前窗口,连并NavigationBar也一并遮住的话,那么你需要 将视图添加到navigationController的View上 [self.navigati ...

  2. SqlLocalDB使用笔记

    一.介绍 SqlLocalDB是VS安装时附带的数据库软件,相当于精简版的SQL Express. 二.使用 VS版本为2015,默认安装位置为:C:\Program Files\Microsoft ...

  3. 使用Windows上的Eclipse 远程调试 linux下的Tomcat

    1:修改Linux上Tomcat的catalina.sh,第一行添加declare -x CATALINA_OPTS="-Xdebug -Xrunjdwp:transport=dt_sock ...

  4. php下载网络图片到服务器

    /** * 下载二维码到服务器 * @param string $url 图片路径 * @param string $filestring 要保存的文件名 */    private function ...

  5. [ 转 ] scrapy 中解决 xpath 中的中文编码问题

    1.问题描述: 实现定位<h2>品牌</h2>节点 brand_tag = sel.xpath("//h2[text()= '品牌']") 报错:Value ...

  6. WinForm窗体嵌入

    一.在winform窗体上添加两个控件 1.容器>Panel 2.添加 SideBar.dll (下载链接:http://pan.baidu.com/s/1o6qhf9w) (1)将SideBa ...

  7. winform开发之UI系列

    1.如何构造一个漂亮的主窗体 主要讲述如何对一个新建窗体的美化过程,涉及到经常需要用到的几个属性我会着重强调它的用法,并不断更新它,因为楼主也正在探索中.... 步骤如下: vs新建一个winform ...

  8. 构建Maven聚合工程

    最近,准备构建一个Maven的项目,基于模块化的结构思想,决定使用Maven的聚合工程进行构建. 环境准备: (1)eclipse 这里推荐使用spring封装的eclipse(即STS) ,STS中 ...

  9. 配置移动前端开发调试环境(nodejs+npm+weiner的安装和配置使用)

    这段时间发现做移动端的开发调试是一大难题,网上逛了逛发现有一些工具可用,如chrome的远程调试,实际测试过程中我始终调试不成功,听说被墙后是不行的,所以最终找了如下的方法. 因为基于nodeJS环境 ...

  10. Java数组的一些基本算法

    数组的一些算法问题:  排序:(升序)   选择排序:     求每一轮的最小值:再输出   冒泡排序:     相邻的两个数相比较,把两个数相比较,第一个大于好面的就交换位置   shell排序: ...