AngularJS:Direcitve指令用法解读(上)

本篇转载于:http://blog.csdn.net/evankaka
AngularJs GitHub: https://github.com/angular/angular.js/
AngularJs下载地址:https://angularjs.org/
一、Directive的定义及其使用方法
大概如下:

angular.module("app",[]).direcitve("directiveName",function(){
   return{
       //通过设置项来定义
   }
})

Directive可以设置于元素名、属性、class、注释中。下面引用myDir这个directive的等价方式。(很多情况下directive都限制为"属性"的使用方式)

<!DOCTYPE html>  
<html lang="zh" ng-app="myApp">  
<head>  
    <meta charset="UTF-8">  
    <title>AngularJS入门学习</title>  
    <script type="text/javascript" src="./1.5.3/angular.min.js"></script>  
</head>  
<body> 
    <hello-world></hello-world> 
</body>
<script type="text/javascript">
  var app=angular.module('myApp',[]);
app.directive('helloworld',function(){
    return{
        restrict::'E',
        template:'<div>Hi,我是林炳文~~~~</div>',
       replace:true
  };
});
</script>
</html>

结果:

image.png

二、Direcitve指令内容解读
1、restrict(字符串)可选参数,指明指令在DOMDOM里面以什么形式被声明。取值有:

E(元素):<directiveName></directiveName>
A(属性):<div directiveName="expression"></div>
C(类):<div class="directiveNmae"></div>
M(注释):<--directive:directiveName expression>

一般情况下E/A用的比较多。
2、priority(数字),可选参数,指明指令的优先级。若在单个DOM上有多个指令。
3、template(布尔型),可选参数,可被设置为true或false,若设置为true,则优先级低于此指令的其他指令无效。不会被调用。(优先级相同的还会执行)。
4、template(字符串或者函数)可选参数,可以是:
(1)一段HTML文本

<!DOCTYPE html>  
<html lang="zh" ng-app="myApp">  
<head>  
  <meta charset="UTF-8">  
  <title>AngularJS入门学习</title>  
  <script type="text/javascript" src="./1.5.3/angular.min.js"></script>  
</head> 
<body> 
  <hello-world></hello-world> 
</body>
<script type="text/javascript">
  var app=angular.module('myApp',[]);
  app.directive('helloworld',function(){
      return {
            restrict:'E',
            template:'<div><h1>Hi. 我是林炳文~~~</h1></div>'
      }    
})
</script>

结果:


image.png

(2)一个函数,可接受两个参数tElement和tAttrs
其中tElement是指使用指令的元素,而tAttrs则实例的属性。它是由一个元素上所有的属性组成的集合(对象)形如:

<hello-world2 title='我是第二个directive'> </hello-world2>

其中title就是在tAttrs上的属性
下面是template 是一个函数的情况

<!DOCTYPE html>  
<html lang="zh" ng-app="myApp">  
<head>  
    <meta charset="UTF-8">  
    <title>AngularJS入门学习</title>  
    <script type="text/javascript" src="./1.5.3/angular.min.js"></script>  
</head>  
<body>  
<hello-world></hello-world>  
<hello-world2 title="我是第二个directive"></hello-world2>
</body>
<script type="javascript">
  var app=angulat.module('myApp',[]);
  app.directive('helloworld',funtion(){
       return{
          restrict:'E',  
          template:'<div><h1>Hi,我是林炳文~~~</h1></div>',
          replace:true
      };
  });
app.directive('helloworld2',function(){
      return{
          restrict:'EAC',
          template:function(tElement,tAttrs){
              var _html="";
              _html+='<div>'+'hello'+tAttrs.title+'</div>';
              return  _html;
           }     
      };
});
</script>

结果:


image.png

5、templateUrl(字符串或者函数),可选参数,可以是
(1)一个代表HTML文件路径的字符串。
(2)一个函数,可接受两个参数tElement和tAttrs。
注意:在本地开发的时候。需要运行一个服务器,不然适用templateUrl会报错。由于加载HTML模板是通过异步的方式加载的,若加载大量的模板会拖慢网站的加载速度。有个技巧,就是缓存模板。

<script type="text/ng-template" id="hello.html">
    <div><h1>Hi,我是林炳文~~~</h1></div>
</script>
</html>
这里的```id```属性就是被设置在templateUrl上用的。

实例:

<!DOCTYPE html>  
<html lang="zh" ng-app="myApp">  
<head>  
    <meta charset="UTF-8">  
    <title>AngularJS入门学习</title>  
    <script type="text/javascript" src="./1.5.3/angular.min.js"></script>  
</head>  
<body>  
<hello-world></hello-world>  
</body>  
<script type="text/javascript">  
    var app=angular.module('myApp',[]);
    app.directive('helloworld',function(){
         return {  
             restrict: 'E',  
            templateUrl: 'hello.html',  
           replace: true  
        };  
   })
</script>
<script type="text/ng-template" id="hello.html">
    <div><h1>Hi,我是林炳文~~~</h1></div>
</script>
</html>
这里的```id```属性就是被设置在templateUrl上用的。

结果:


image.png

还有一种缓存办法是:

app.run(["$templateCache",function($templateCache){
    $templateCache.put('hello.html',"  <div><h1>Hi,我是林炳文~~~</h1></div>");
}]);

使用实例:

<!DOCTYPE html>  
<html lang="zh" ng-app="myApp">  
<head>  
   <meta charset="UTF-8">  
   <title>AngularJS入门学习</title>  
   <script type="text/javascript" src="./1.5.3/angular.min.js"></script>  
</head>  
<body>  
<hello-world></hello-world>  
</body>  
<script type="text/javascript">  
var app = angular.module('myApp', []);  
app.directive('helloWorld', function() {  
   return {  
       restrict: 'E',  
       templateUrl: 'hello.html',  
       replace: true  
   };  
});
app.run(["$templateCache",function($templateCache){
   $templateCache.put('hello.html',' <div><h1>Hi,我是林炳文~~~</h1></div>');
}]);
</script>  
</html>  

结果:

image.png

纵观觉得还是第一种方法比较好,写起来快。直接包在script当中。
6.replace (布尔值),默认为false,设置为true的时候,看下面的例子。
image.png

image.png

replacetrue的时候,hello-world这个标签不在了,反之,则存在。
7、scope
(1)默认值false。表示继承父作用域;
(2)true。表示继承父作用域,并创建自己的作用域(子作用域);
(3){}。表示创建创建一个全新的隔离作用域。
7.1实例:scope的继承机制。利用ng-controller这个指令来举例。

<!DOCTYPE html>  
<html lang="zh" ng-app="myApp">  
<head>  
    <meta charset="UTF-8">  
    <title>AngularJS入门学习</title>  
    <script type="text/javascript" src="./1.5.3/angular.min.js"></script>  
</head>  
<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>  
</html>  

scope:false


image.gif

scope:true


image.gif

scope:{}
iamges.gif

当为false的时候,儿子继承父亲的值,改变父亲的值,儿子的值也会随之变化,反之亦然。(继承不隔离)
当为true的时候,儿子继承父亲的值,改变父亲的值,儿子的值随之改变,但是改变儿子的值,父亲的值不受影响(继承隔离)。
当为{}的时候,没有继承父亲的值,所以儿子的值为空,改变任何一方都不会影响另一方的值。(不继承隔离)。
7.2隔离作用域可以通过绑定策略来访问父作用域的属性。
directive在使用隔离scope的时候,提供了了三种方法同隔离之处的地方交互,这三种分别是:
①@绑定一个局部scope属性当前dom节点的属性值。结果总是一个字符串,因为DOM的属性是字符串。
②&提供一种方法执行一个表达式在父scope的上下文中,如果没有指定的attr名称,则属性名称为相同的本地名称。
③=通过directive的attr属性的值在局部scope的属性和父scope属性名之间建立双向绑定。
@局部scope属性
@方式局部属性用来访问directive外部环境定义的字符串值,主要是通过directive所在的标签属性绑定外部字符串的值,这种绑定是单向的,即父scope的绑定变化,directive中的scope的属性会同步变化,而隔离scope中的绑定变化,父scope是不知道的。
如下实例:directive声明未隔离scope类型,并且使用@绑定name属性,在directive中使用name绑定父scope中的属性,当改变scope中属性的值的时候,directive会同步更新值,当改变directive的scope的属性值,父scope无法同步更新值。
js代码:

<!DOCTYPE html>  
<html lang="zh" ng-app="myApp">  
<head>  
    <meta charset="UTF-8">  
    <title>AngularJS入门学习</title>  
    <script type="text/javascript" src="./1.5.3/angular.min.js"></script>  
</head>  
<body>
 <div ng-controller="myController">
      <div class="result">
         <div>父scope
            <div>Say:{{name}}<br>改变父scope的name:<input type="text" vaule="" 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>  
</body>
<script type="text/javascript">
    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">'
        };
    })
</script>
</html>

结果:页面初始结果

image.png

动画效果:
image.gif

可以看到父scope上的内容发生了改变,子scope同时发生改变,而子scope上的内容发生改变,不影响父scope上的内容。

=局部scope属性
=通过directive的attr属性的值在局部scope的属性和父scope属性名之间建立双向绑定。
意思是,当你想要一个双向绑定的时候,可以使用=来引入外部属性,无论是改变父scope还是隔离scope里的属性,父scope和隔离scope都会同时更新属性值,因为它们是双向绑定的关系。
示例代码:

<!DOCTYPE html>  
<html lang="zh" ng-app="myApp">  
<head>  
    <meta charset="UTF-8">  
    <title>AngularJS入门学习</title>  
    <script type="text/javascript" src="./1.5.3/angular.min.js"></script>  
</head>  
<body>  
<div ng-controller="myController">  
    <div>父scope:  
        <div>Say:{{user.name}}<br>改变父scope的name:<input type="text" value="" ng-model="userBase.name"/></div>  
    </div>  
    <div>隔离scope:  
        <div isolated-directive user="userBase"></div>  
    </div>  
</div>  
</body>  
<script type="text/javascript">  
       var app=angular.module('myApp',[]);
       app.controller('myController',function ($scope){
          $scope.userBase={
              name:'hello',
              id:1
          }
    }).directive('isolateDirective',function(){
        return{
            scope:{
              user:"="
            },
          template:'Say:{{user.name}}<br>改变隔离scope的name<input type="button" value="" ng-model="user.name">'
        }
    })
</script>
</html>

结果:

image.gif

可以看到父scope和子scope上的内容一直都是一样的!

& 局部 scope 属性
& 方式提供一种途经是 directive 能在父 scope 的上下文中执行一个表达式。此表达式可以是一个 function。
比如当你写了一个 directive,当用户点击按钮时,directive 想要通知 controller,controller 无法知道 directive 中发生了什么,也许你可以通过使用 angular 中的 event 广播来做到,但是必须要在 controller 中增加一个事件监听方法。
最好的方法就是让 directive 可以通过一个父 scope 中的 function,当 directive 中有什么动作需要更新到父 scope 中的时候,可以在父 scope 上下文中执行一段代码或者一个函数。

如下示例在 directive 中执行父 scope 的 function。

<!DOCTYPE html>  
<html lang="zh" ng-app="myApp">  
<head>  
  <meta charset="UTF-8">  
  <title>AngularJS入门学习</title>  
  <script type="text/javascript" src="./1.5.3/angular.min.js"></script>  
</head>  
<body>  
<div  ng-controller="myController">  
      <div>父scope:  
          <div>Say:{{value}}</div>  
      </div>  
      <div>隔离scope:  
          <div isolated-directive action="click()"></div>  
      </div>  
</div>  
</body>  
<script type="text/javascript">  
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()"/>'  
      }  
  })  
</script>  
</html>  
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,463评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,868评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,213评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,666评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,759评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,725评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,716评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,484评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,928评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,233评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,393评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,073评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,718评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,308评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,538评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,338评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,260评论 2 352