学了前面这么多关于指令的知识,现在就用指令来写一个小组件:expander

这个组件的功能就是点击开展菜单,再点击收起菜单:

↑↓点击展开收起

下面来看它的代码:

html:

<!DOCTYPE html>
<html ng-app="expanderModule">
<head>
<title>20.9 指令-expander</title>
<meta charset="utf-8">
<link href="../bootstrap.css" rel="stylesheet">
<script src="../angular.min.js"></script>
<script type="text/ng-template" id="text.html">
<div class="btn-group">
<button class="btn btn-primary btn-sm dropdown-toggle" type="button" data-toggle="dropdown" ng-click="toggle()">
{{title}}<span class="caret"></span>
</button>
<ul class="dropdown-menu" ng-show="ifShow" ng-transclude>
</ul>
</div>
</script>
<script src="script.js"></script>
<style type="text/css">
body{
padding:40px
}
</style>
</head>
<body>
<div ng-controller="expanderCtrl">
<expander my-title="{{title}}">
<li ng-repeat="content in contents">
<a href="">{{content}}</a>
</li>
</expander>
</div>
</body>
</html>

js:

/*20.9 指令*/
var expanderModule = angular.module('expanderModule',[]);
expanderModule.controller('expanderCtrl',function($scope){
$scope.title = "标题";
$scope.contents = ['bunny','cat','dog'];
});
expanderModule.directive('expander',function(){
return {
restrict:'EA',
replace:true,
templateUrl:'text.html',
transclude:true,
scope:{title:'@myTitle'},
link:function(scope,ele,attrs){
scope.ifShow = false;
scope.toggle = function(){
scope.ifShow = !scope.ifShow;
}
}
}
});
//指令元素里原本就包含的内容,他的作用域使用元素所在的作用域的父作用域,而不是指令的独立scope,所以contents值可以访问到控制器的contents,但是template里面的值却是访问指令独立作用域的.
//这里的myTitle不会被修改,所以采用@和=绑定都可以

下面来分析一下这个指令:

1.创建一个名叫expander的指令,它的类型是元素

2.指令元素里面放置的是子菜单的内容,注意是内容,不是整个ul.因为它要作为一坨插入到ng-transclude元素中,所以,ul必须充当ng-transclude这个元素的角色,然后设置transclue:true.

3.指令的my-title属性用于绑定父作用域的title属性.在这个案例中,使用@绑定和使用=绑定都是可以的.

4.在link函数里给指令独立scope添加ifShow属性.然后在ul子菜单里通过ng-show="ifShow"来绑定它的显示与隐藏

5.在link函数里给指令独立scope添加toggle()方法,切换ifShow.然后给按钮绑定ng-click="toggle()",来改变ifShow的值.

6.在控制器里添加相关的数据模型:title和contents

*7.这一点很重要:

我在expander指令元素里面写了:ng-repeat="content in contents".

然后在text.html模板里写了:ng-click="toggle()" , {{title}} , ng-show="ifShow".

  对于expander指令元素来说,写在它里面的内容,它的作用域就是指令所在的元素的作用域,也就是父作用域,所以它可以访问到控制器里的contents模型.

  但是text.html,作为指令的模板,被替换到指令,写在模板里面的内容,它的作用域就是指令的独立作用域.也就是指令里定义的scope,所以它是不能访问到contents的,而toggle方法,ifShow属性,都已经在写link函数时,通过scope参数给指令独立作用域绑定了这两个属性和方法.title属性也通过@策略绑定父作用域的title属性.

总结来说就是:原来就写在指令里面的内容,它的作用域还是父作用域,和指令的scope无关.

指令的template里面的内容,它的作用域是指令的作用域,就是定义指令时候写的scope属性

完整代码:https://github.com/OOP-Code-Bunny/angular/blob/master/OREILLY/20.9%20%E6%8C%87%E4%BB%A4.html

https://github.com/OOP-Code-Bunny/angular/blob/master/OREILLY/script.js

angular学习笔记(三十)-指令(9)-一个简单的指令示例的更多相关文章

  1. angular学习笔记(三十)-指令(10)-require和controller

    本篇介绍指令的最后两个属性,require和controller 当一个指令需要和父元素指令进行通信的时候,它们就会用到这两个属性,什么意思还是要看栗子: html: <outer‐direct ...

  2. angular学习笔记(三十)-指令(7)-compile和link(1)

    这篇主要讲解指令中的compile,以及它和link的微妙的关系. link函数在之前已经讲过了,而compile函数,它和link函数是不能共存的,如果定义了compile属性又定义link属性,那 ...

  3. angular学习笔记(三十)-指令(6)-transclude()方法(又称linker()方法)-模拟ng-repeat指令

    在angular学习笔记(三十)-指令(4)-transclude文章的末尾提到了,如果在指令中需要反复使用被嵌套的那一坨,需要使用transclude()方法. 在angular学习笔记(三十)-指 ...

  4. angular学习笔记(三十)-指令(5)-link

    这篇主要介绍angular指令中的link属性: link:function(scope,iEle,iAttrs,ctrl,linker){ .... } link属性值为一个函数,这个函数有五个参数 ...

  5. angular学习笔记(三十)-指令(2)-restrice,replace,template

    本篇主要讲解指令中的 restrict属性, replace属性, template属性 这三个属性 一. restrict: 字符串.定义指令在视图中的使用方式,一共有四种使用方式: 1. 元素: ...

  6. angular学习笔记(三十)-指令(7)-compile和link(2)

    继续上一篇:angular学习笔记(三十)-指令(7)-compile和link(1) 上一篇讲了compile函数的基本概念,接下来详细讲解compile和link的执行顺序. 看一段三个指令嵌套的 ...

  7. angular学习笔记(三十)-指令(1)-概述

    之前在 angular学习笔记(十九)-指令修改dom 里面已经简单的提到了angular中的指令,现在来详细的介绍 '指令' 一.指令的创建: dirAppModule.directive('dir ...

  8. angular学习笔记(三十)-指令(8)-scope

    本篇讲解指令的scope属性: scope属性值可以有三种: 一.scope:false 默认值,这种情况下,指令的作用域就是指令元素当前所在的作用域. 二.scope:true 创建一个继承了父作用 ...

  9. angular学习笔记(三十)-指令(7)-compile和link(3)

    本篇接着上一篇来讲解当指令中带有template(templateUrl)时,compile和link的执行顺序: 把上一个例子的代码再进行一些修改: 1.将level-two指令改成具有templa ...

随机推荐

  1. RHEL7 添加网卡并设置多个IP地址

    首先在虚拟机管理界面中给该虚拟机添加一块虚拟网卡,VirtualBox虚拟机中要shutdown才能添加. [root@rhel7 ~]# nmcli device show | grep -i de ...

  2. Spark -14:spark Hadoop 高可用模式下读写hdfs

    第一种,通过配置文件 val sc = new SparkContext() sc.hadoopConfiguration.set("fs.defaultFS", "hd ...

  3. Ubuntu 如何downgrade降级系统

    大家都熟悉通过网络upgrade升级Linux系统,这里要介绍的是如何downgrade降级系统.背景情况是这样的:我将系统升级到feisty h3之后,发现libvte-0.15.3有个bug,会影 ...

  4. iOS控制器生命周期

    //view加载完毕就会自动调用 - (void)viewDidLoad; //view即将显示到window上 - (void)viewWillAppear:(BOOL)animated ; //v ...

  5. 【DeepLearning】Exercise:PCA in 2D

    Exercise:PCA in 2D 习题的链接:Exercise:PCA in 2D pca_2d.m close all %%=================================== ...

  6. 【LeetCode】Missing Ranges

    Missing Ranges Given a sorted integer array where the range of elements are [lower, upper] inclusive ...

  7. syslog远程日志存储/514端口【转】

    昨天在抓包的时候,发现在514端口,有SYSLOG字段的东西,不知道是用来干啥的,现在来分析一下: 其实他是在电脑间用了syslog远程日志存储,他用udp监控了514端口的数据流,之后收集整理日志: ...

  8. LNMP分离式部署实例[转]

    很多人在练习部署LNMP环境的时候,大都数是部署在同一个虚拟机上面的.但是实际工作中,我们一般都是分离部署的. 今天我就用3台虚拟机,部署下LNMP环境.以供参考! 网络拓扑图: 首先准备3台虚拟机: ...

  9. .net获取当前url各种属性(文件名、参数、域名、端口等)的方法(转)

    假设当前页完整地址是:http://www.test.com:80/aaa/bbb.aspx?id=5&name=kelli "http://"是协议名 "www ...

  10. Win10下Clion配置opencv3

    本人不想在爱机装一个vs2013或者vs2015这种庞然大物,可是手头要弄一个基于windows的opencv项目,就只好装了个Clion,期间踩了不少坑,记录一下. 参考网址:http://www. ...