什么是Directive

Directive将一段html,js封装在一起,形成一个可以复用的独立个体,具有特定的功能。angularjs中的指令通常是比较小的组件,它相当于是给我们提供了一些公共的自定义的DOM元素、class属性或attr属性。除此之外,我们可以在这个基础上来操作scope,绑定事件,更改样式等等。通过Directive我们可以封装很多公共指令,比如分页、自动补全等多个指令,封装好后,我们下次复用只要在html页面上添加该指令就可以实现复杂的功能。 
总结一下使用Directive的场景: 
1.抽象一个自定义组件,在其他地方重用。 
2.使html更具语义化,不需要深入研究代码和逻辑就可以知道页面实现了哪些功能。 
angular封装了很多自己的指令,都是ng开头,比如ng-model,ng-controller,ng-show,ng-click等等,我们也可以封装自己的指令,在当前的节点上干点特别的事。

 

例子

html部分:

<body ng-app="testApp">
<div ng-controller="testController">
<input type="text" ng-model="city" placeholder="Enter a city" />
<my-test ng-model="city" ></my-test>
<span my-test="exp" ng-model="city"></span>
<span ng-model="city"></span>
</div>
</body>

js部分

var app = angular.module('testApp', [
'directives',
'controllers'
]); // 自定义directive
var myDirective = angular.modeule('directives', []);
myDirective.directive('myTest', function() {
return {
restrict: 'EMAC',
require: '^ngModel',
scope: {
ngModel: '='
},
template: '<div><h4>Weather for {{ngModel}}</h4</div>'
};
}); // 定义controller
var myControllers = angular.module('controllers', []);
myControllers.controller('testController', [
'$scope',
function($scope) {
$scope.name = 'this is directive1';
}
]);

以上是一个自定义myTest这个directive的应用。来看一下定义一个directive可以用哪些格式和参数:

var myDirective = angular.module('directives', []);

myDirective.directive('directiveName', function() {
return {
template: '<div></div>',
replace: false,
transclude: true,
restrict: 'E',
scope: {},
controller: function($scope, $element) { },
complie: function(tElement, tAttrs, transclude) {
return {
pre: function preLink(scope, iElement, iAttrs, controller) { },
post: function postLink(scope, iElement, iAttrs, controller) { }
};
},
link: function(scope, iElement, iAttrs) { }
};
});

我们可以发现directive最关键的部分就是通过return来设置参数,那么return里可以设置哪些参数呢,下面具体讲一讲这些参数及其说明。 
1.name 
当前scope的名称,一般声明时使用默认值,不用手动设置此属性。 
2.priority 
优先级。当有多个directive定义在同一个DOM元素上时,有时要明确他们的执行顺序。如果优先级相同,那么执行顺序不确定(一般情况下,优先级高的先执行,相同优先级的按照先绑定后执行)。 
3.teminal 
最后一组。如果设置为true,那么当前的priority将会成为最后一组执行的directive。也就是说,比这个directive的priority更低的directive将不会执行,同优先级的依然会执行,但执行顺序不确定。 
4.scope 
(1)true 
表示为这个directive创建一个新的scope。如果在同一个元素中有多个directive需要新的scope的话,它还是只会创建一个scope。 
(2){} 
将会创建一个新的,独立的scope,这个scope与一般的scope的区别在于它不是通过原型继承父scope的。这对可复用的组件而言很有用,可以有效的防止读取或者修改父级scope的数据。 
5.controller 
controller允许其他directive通过指定名称的require进行共享,也就是说directive之间可以相互沟通,增强相互之间的行为。controller默认注入以下本地对象: 
$scope:与当前元素结合的scope 
$element:当前元素 
$attrs:当前元素的属性对象 
$transclude:一个预先绑定到当前scope的转置linking function 
6.require 
请求另外的controller,传入当前directive的linking function中,require需要传入一个directive controller的名称,如果没有对应的controller会抛出一个error。名称可以加入以下前缀: 
?: 不要抛出异常,这就使得这个依赖变为一个可选项 
^: 允许查找父元素的controller

7.restrict 
EACM的子集的字符串,限制了directive为指定的声明方式。省略的话,directive将紧紧允许通过属性声明的方式: 
E:元素 
A:属性 
C:class名 
M:注释 
比较常用的是EA 
8.template 
如果replace为true,则将模板内容替换当前的html元素,并将原来元素的属性,class一并转移;如果replace为false,则将模板元素作为当前元素的子元素。 
9.templateUrl 
与template基本一致,但模板通过指定的url进行加载。由于模板加载是异步的,所有compilation,linking都会暂停,等加载完毕后再执行。 
10.replace 
设置为true,那么模板将会替换当前元素,而不是作为子元素添加到当前元素中。(为true时模板必须要有一个根节点) 
11.transclude 
编译元素的内容,使它能够被directive使用。需要在模板中配合ngTransclude使用。 
12.compile 
13.link

看一个transclude的例子

<body>
<hello>
<br/><span>原始的内容,</span><br/>
<span>还会在这里。</span>
</hello>
<hello>
</hello>
</body>
var appModule = angular.module('app', []);
appModule.directive('hello', function() {
return {
restrict: 'E',
template: '<div>Hi there <span ng-transclude></span></div>',
transclude: true
};
});

可以发现,这里的ng-transclude会把hello标签里内容读取出来,如果transclude设置为false,那么浏览器就不会处理hello这个标签里内容。 
注意:如果指令使用了transclude参数,那么控制器就无法正常监听数据模型的变化。建议在链接函数里使用$watch服务。

接下来重点讲一下scope。 
scope主要是用来隔离作用域的,那么scope在true,false,{}时值是如何变化的呢,看下面一个例子:

<body>
<div ng-controller='MainController'>
父亲:{{name}}<input ng-model="name" />
<div my-directive></div>
</div>
</body>
<script type="text/javascript">
var app = angular.module('myApp', []);
app.controller('MainController', function ($scope) {
$scope.name = '林炳文';
});
app.directive('myDirective', function () {
return {
restrict: 'EA',
scope:false,
template: '<div>儿子:{{ name }}<input ng-model="name"/></div>'
};
});
</script>

scope:false时,子scope继承父scope的值,改变父亲的值,儿子的值也会改变,反之一样(继承不隔离) 
 
scope:true时,子scope继承父scope的值,改变父亲的值,儿子的值也会改变。但是改变儿子的值,父亲的值不变(继承隔离) 
 
scope:{}时,没有继承父scope,改变任何一方不会改变另一方的值(不继承隔离) 

当你想要创建一个可重用的组件时,隔离作用域是一个很好的选择,这样可以防止父作用域被污染。那么隔离作用域后如何访问父级作用域呢?

angular通过绑定策略来访问父作用域的属性。 
directive提供了三种方法同隔离之外的地方进行交互: 
(1) @ 
主要通过directive所在的标签属性绑定外部字符串值。这种绑定是单向的,即父scope变化,directive中的scope的对应属性也会变化。但是隔离scope中的绑定变化,父scope是不知道的。

<div ng-controller="myController">
<div class="result">
<div>父scope:
<div>Say:{{name}}<br>改变父scope的name:<input type="text" value="" ng-model="name"/></div>
</div>
<div>隔离scope:
<div isolated-directive name="{{name}}"></div>
</div>
<div>隔离scope(不使用父scope {{name}}):
<div isolated-directive name="name"></div>
</div>
</div>
</div>
var app = angular.module('myApp', []);
app.controller("myController", function ($scope) {
$scope.name = "hello world";
}).directive("isolatedDirective", function () {
return {
scope: {
name: "@"
},
template: 'Say:{{name}} <br>改变隔离scope的name:<input type="buttom" value="" ng-model="name" class="ng-pristine ng-valid">'
};
});

 
(2) = 
通过directive的attr属性在局部scope和父scope的属性名之间建立双向绑定 
(3) & 
directive在父scope的上下文中执行一个表达式,此表达式是一个function。directive可以通过一个父scope中的function,当directive中有什么动作需要更新到父scope中的时候,可以在父scope上下文中执行一段代码或一个函数。

<div  ng-controller="myController">
<div>父scope:
<div>Say:{{value}}</div>
</div>
<div>隔离scope:
<div isolated-directive action="click()"></div>
</div>
</div>
var app = angular.module('myApp', []);
app.controller("myController", function ($scope) {
$scope.value = "hello world";
$scope.click = function () {
$scope.value = Math.random();
};
}).directive("isolatedDirective", function () {
return {
scope: {
action: "&"
},
template: '<input type="button" value="在directive中执行父scope定义的方法" ng-click="action()"/>'
}
})

 
再看一个scope应用的例子

<body ng-app="testApp">
<div ng-controller="attrtest">
<my-attr info="naomi"></my-attr>
</div>
</body>
myDirectives.directive('myAttr', function() {
return {
restrict: 'E',
scope: {
customerInfo: '=info'
},
template: 'Name: {{customerInfo.name}} Address: {{customerInfo.address}}<br>' +
'Name: {{vojta.name}} Address: {{vojta.address}}'
};
}); myControllers.controller('attrtest',['$scope',
function($scope) {
$scope.naomi = {
name: 'Naomi',
address: '1600 Amphitheatre'
};
$scope.vojta = {
name: 'Vojta',
address: '3456 Somewhere Else'
};
}
]);

结果: 
Name: Address: 
Name: Vojta Address: 3456 Somewhere Else

接下来讲一下require

<body>
<outer-directive>
<inner-directive></inner-directive>
<inner-directive2></inner-directive2>
</outer-directive>
<script>
var app = angular.module('myApp', []);
app.directive('outerDirective', function() {
return {
scope: {},
restrict: 'AE',
controller: function($scope) {
this.say = function(someDirective) {
console.log('Got:' + someDirective.message);
};
}
};
});
app.directive('innerDirective', function() {
return {
scope: {},
restrict: 'AE',
require: '^outerDirective',
link: function(scope, elem, attrs, controllerInstance) {
scope.message = "Hi,leifeng";
controllerInstance.say(scope);
}
};
});
app.directive('innerDirective2', function() {
return {
scope: {},
restrict: 'AE',
require: '^outerDirective',
link: function(scope, elem, attrs, controllerInstance) {
scope.message = "Hi,shushu";
controllerInstance.say(scope);
}
};
}); </script> </body>

结果:
Got:Hi,leifeng
Got:Hi,shushu
上例中的innerDirective和innerDirective2复用了outerDirective的controller中的方法。这也进一步说明,指令中的controller是用来让不同指令间通信用的。

require的值是另一个指令的名称,但实际上引用的是那个指令的控制器实例。require非常有用,因为很多时候指令之间是要相互配合的,比如说require:'ngModel',那么当angular初始化它的时候,就会在它所在的元素上寻找一个叫做ng-model的指令,然后取得它的控制器实例。找到时候,就可以把这个控制器的实例作为Link函数的第四个参数传进来。 
关于require:'ngModel'其他的一些用法,会在directive二讲到。

再说说controller,link,compile之间的关系

myController.controller('directive2',[
'$scope',
function($scope) {
$scope.number = '1111 ';
}
]); myDirective.directive('exampleDirective', function() {
return {
restrict: 'E',
template: '<p>Hello {{number}}!</p>',
controller: function($scope, $element){
$scope.number = $scope.number + "22222 ";
},
link: function(scope, el, attr) {
scope.number = scope.number + "33333 ";
},
compile: function(element, attributes) {
return {
pre: function preLink(scope, element, attributes) {
scope.number = scope.number + "44444 ";
},
post: function postLink(scope, element, attributes) {
scope.number = scope.number + "55555 ";
}
};
}
}
});
<body ng-app="testApp">
<div ng-controller="directive2">
<example-directive></example-directive>
</div>
</body>

看一下结果: 
Hello 1111 22222 44444 5555 ! 
从结果可以看出,controller先运行,compile后运行,link不运行。 
注掉compile,结果是: 
Hello 1111 22222 33333 ! 
这次是controller先运行,link后运行。 
也就是说,link和compile不兼容,一般的,compile比link的优先级要高。

注意,在controller和link中都可以定义方法,它们的区别是,controller主要是用来提供可在指令间复用的行为,但link链接函数只能在当前内部指令中定义行为,无法再指令复用。

directive 指令一的更多相关文章

  1. 学习AngularJs:Directive指令用法

    跟我学AngularJs:Directive指令用法解读(上) http://blog.csdn.net/evankaka/article/details/51232895 跟我学AngularJs: ...

  2. 学习AngularJs:Directive指令用法(完整版)

    这篇文章主要学习AngularJs:Directive指令用法,内容很全面,感兴趣的小伙伴们可以参考一下   本教程使用AngularJs版本:1.5.3 AngularJs GitHub: http ...

  3. angular上传获取图片的directive指令

    在AngularJS中,操作DOM一般在指令中完成,那么指令是如何实现的呢?指令的作用是把我们自定义的语义化标签替换成浏览器能够认识的HTML标签 一般的事件监听是在对静态的dom绑定事件,而如果在指 ...

  4. AngularJs:Directive指令用法

    摘自:http://www.jb51.net/article/83051.htm 摘要:Directive(指令)是AngularJ非常强大而有有用的功能之一.它就相当于为我们写了公共的自定义DOM元 ...

  5. AngularJS中Directive指令系列 - 基本用法

    参考: https://docs.angularjs.org/api/ng/service/$compile http://www.zouyesheng.com/angular.html Direct ...

  6. 关于Angularjs写directive指令传递参数

    包子又来啦.... 在Angularjs当中,我们可能会经常要写directive指令.但是指令如果要共用的话,肯定是有细微的差别的,所以这些差别可能需要一个参数来决定 所以如何在指令中传递参数呢.. ...

  7. angularjs中directive指令与component组件有什么区别?

     壹 ❀ 引 我在前面花了两篇博客分别系统化介绍了angularjs中的directive指令与component组件,当然directive也能实现组件这点毋庸置疑.在了解完两者后,即便我们知道co ...

  8. AngularJS中Directive指令系列 - scope属性的使用

    文章是转的,我做下补充.原文地址:https://segmentfault.com/a/1190000002773689 每当一个指令被创建的时候,都会有这样一个选择,是继承自己的父作用域(一般是外部 ...

  9. AngularJS directive 指令相关记录

    .... .directive('scopeDemo',function(){ return{ template: "<div class='panel-body'>Name: ...

  10. angular源码分析 摘抄 王大鹏 博客 directive指令及系列

    链接地址:http://www.cnblogs.com/web2-developer/p/angular-14.html $compile的功能:将一个html字符串或者一个DOM进行编译,返回一个链 ...

随机推荐

  1. nDPI 的论文阅读和机制解析

    nDPI: Open-Source High-Speed Deep Packet Inspection Wireless Communications & Mobile Computing C ...

  2. 使用mongodb提供的dotnet core sdk进行地理位置运算

    mongodb提供地理位置运算功能,比较常用的场景比如,先判断用户所在的街道,然后看看街道附近有啥餐厅,然后算算用户与餐厅的距离什么的,官网里提供了比较详细的demo介绍不同api的用法 此处记录下d ...

  3. web相关文件加载顺序

    WEB相关文件的加载顺序   一. 1.启动一个WEB项目,WEB容器会先去读取它的配置文件web.xml,读取<context-param>和<listener>两个节点. ...

  4. Java并发编程(九)线程间协作(下)

    上篇我们讲了使用wait()和notify()使线程间实现合作,这种方式很直接也很灵活,但是使用之前需要获取对象的锁,notify()调用的次数如果小于等待线程的数量就会导致有的线程会一直等待下去.这 ...

  5. nginx详解反向代理、负载均衡、LNMP架构上线动态网站

    简介 Nginx是俄罗斯人编写的十分轻量级的HTTP服务器,Nginx,它的发音为“engine X”,是一个高性能的HTTP和反向代理服务器,同时也是一个IMAP/POP3/SMTP 代理服务器.N ...

  6. 浅谈nodejs中HTTP模块应用

    这里给大家分享下后端人员如果利用nodejs对数据的一些处理情况  适用于初学者使用 大牛勿喷 给大家分享下主要后端思想部分代码,前端部分就不展示了 const http = require(&quo ...

  7. 初次使用vue-cli3 来搭建项目

    1,细数项目中使用的技术:vue, vue-router, vuex ,axios,vue-cli3, 快速建站. 2,mock技术使用的express-mockjs . 由于cli3 最新版的话缺少 ...

  8. filter 图片滤镜的各种设置

    filter 图片滤镜 给当前元素加滤镜_改变它的明亮度 定义:filter 属性定义了元素(通常是<img>)的可视效果(例如:模糊与饱和度).作用在图片上或元素上.div{ },或 d ...

  9. PHP程序员的技术成长规划 第二阶段:提高阶段

    第二阶段:提高阶段 (中级PHP程序员) 重点:提高针对LNMP的技能,能够更全面的对LNMP有熟练的应用.目标:能够随时随地搭建好LNMP环境,快速完成常规配置:能够追查解决大部分遇到的开发和线上环 ...

  10. 嵌入式C语言自我修养 07:地址对齐那些事儿

    7.1 属性声明:aligned GNU C 通过 __atttribute__ 来声明 aligned 和 packed 属性,指定一个变量或类型的对齐方式.这两个属性用来告诉编译器:在给变量分配存 ...