AngularJs中,如何在父元素中调用子元素为自定义Directive中定义的函数?
最近一段时间准备使用AngularJs中的自定义Directive重构一下代码。
在这里说明一下,把自定义控件封装成Directive并不一定是要复用,而是要让代码结构更加清晰。就好像你将一个长方法拆分成多个独立的小方法,也未必要复用它们一样。职责独立等一票好处,会让后期维护更加轻松。
在重构的过程中,我遇到了这样一个问题,先上图:
图一:

这就是我要重构的界面,由于之前时间紧,将这三个Filter和两个button都写在了一个页面中。当时我已经预感到,如果将这里面的状态都写到一个scope上,将出现后期近乎不可维护的灾难。尽管我已经将它们用ViewModel隔离,但代码结构依然让我很不满意。
你说啥?没看出来哪复杂。好,你等着,接下来让你看看里面啥样。
图二:

这是其中一个下拉框展开的模样。什么?你觉得这很一般呐?没看出它多么狰狞?好,你等着。上图三。
图三:

这是三哥展开之后的完全体。现在你应该明白将它们封装成directive是多么必要。
(上图中的这个产品是如今风靡全球的DocAveOnline,使用微软SharePoint的公司没有不知道的。啥?你没听过?偶,忘了说了,只有大公司才用得上)
我重构的思路很简单,将每一个下拉框都独立写成一个directive,然后将哥仨嵌入同一个Directive中,这个外层的directive里面包含Filter和Reset两个Button。
这里原本的交互逻辑是这样的,每个下拉框中的元素都有自己的初始化状态。但是不论你之前怎么霍霍,点击那个Reset按钮之后,要恢复每个下拉框的“出厂设置”。
每个下拉框的Directive拥有独立的scope是肯定的,一切自给自足,与外界无关。也就是说,每个下拉框的reset方法是封装在自己的Directive之中。
那么问题来了,我点击Reset按钮之后,怎么才能调用其子元素中的reset方法?
开始我尝试使用require和link的组合,我是想将reset方法封装到下拉框的controller中,然后在父元素中将三个下拉框对应的directive名字写到require中,这样我就能在父元素的link方法的第四个参数中取到,由这三个子元素controller组成的数组,这我就能轻松调用每个controller的reset方法。(本来controller的作用就是对外暴露API)
可写着写着我就发现不对,好像require只能找自己或者向父元素查找,没法由父元素发起向子元素查找的行动。(也许有,但没找到。如果你知道请在留言中告诉我。)
在江郎才尽的情况下,请教了我师父,他告诉我以下这种方法,其实一点儿都不难,就看你是否理解了自定义directive中,scope里面的,'='这个符号。
稍微懂点儿就知道,这是双向绑定的意思。那解决办法来了,代码如下:
Html部分:
<div ng-controller="myCtrl">
<div>
<inner-directive filter-model="filterModel"></inner-directive>
<button ng-click="filterModel.reset()">click</button>
</div>
</div>
Js部分:
<script>
angular.module('app',[])
.controller('myCtrl',function($scope){
$scope.filterModel = {
name:'xiaomablog'
};
})
.directive('innerDirective',function(){
return {
restrict:'E',
scope:{
filterModel:'='
},
template:'<div ng-bind="filterModel.name"></div>',
link:function(scope,elem,attrs){
scope.filterModel.reset = function(){
console.log('reset');
};
}
}
});
</script>
在innerDirective中的scope属性中,定义了filterModel这个双向绑定的属性,这就意味着directive内部可以霍霍这个属性,外部myCtrl中也可以霍霍这个属性,这样的话彼此都能调用对方霍霍的结果。
我贴心地下了断点,debug给你看:
第一步:
Angular的启动过程是先初始化外部的scope,也就是先执行myCtrl中的方法,给filterModel赋了一个对象;

(注意,此刻filterModel的值为undefined)
第二步:

(没添加reset方法之前,已经看到了之前加的name属性)
然后才执行directive的link方法,将scope与DOM紧紧地联系在一起。
第三步:
此时点击button,已经能调用reset方法了,并在console中打印出'reset'
大家可以尝试在myCtrl和innerDirective的link方法中加断点debug一下,亲自观察执行顺序,加深印象。
注意:
我们了解锤子的方式 , 不是盯着它看 , 而是拿起来用。
—— 马丁·海德格尔
听马丁的话,赶紧去敲。
AngularJs中,如何在父元素中调用子元素为自定义Directive中定义的函数?的更多相关文章
- IE6滤镜在实战测试中能让父层里面的子元素产生阴影
1.写法一: <div class="cornerbg"> <p class="title-file">------</p& ...
- jquery层级原则器(匹配父元素下的子元素)
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ ...
- 【IE6的疯狂之十】父级使用padding后子元素绝对定位的BUG
在前端开发中,经常会用到css的position:absolute来使层浮动,前通过left,top,right等属性来对层进行定位,但ie6对left,top,right等属性的解释和ie7,ie8 ...
- CSS实现父元素半透明,子元素不透明
CSS实现父元素半透明,子元素不透明. 很久以来大家都习惯使用opacity:0.5在新式浏览器里实现半透明,而对IE较旧的版本使用filter:Alpha(opacity=0.5)的滤镜来实现半透明 ...
- 【vue】父组件主动调用子组件 /// 非父子组件传值
一 父组件主动调用子组件: 注意:在父组件使用子组件的标签上注入ref属性,例如: <div id="home"> <v-header ref="he ...
- vue父组件如何调用子组件的属性或方法
常常我们需要组件的拆分,就涉及到父子调用的关系,那么父组件如何调用子组件的属性和方法呢? 子组件child <template> <div> {{msg}} </div& ...
- CSS父元素高度随子元素高度变化而变化
<html> <body> <head> <style> #menu{width:1000px;overfloat:hidden;} /* width: ...
- 父元素设置min-height子元素设置100%问题
问题:父元素设置min-height子元素高度设置100%取不到值,这是因为子元素 div设置 height:100%: 只有当父级元素满足min-height:1000px;设置的条件才触发: 浏览 ...
- AngularJS自定义Directive中link和controller的区别
在AngularJS中,自定义Directive过程中,有时用link和controller都能实现相同的功能.那么,两者有什么区别呢? 使用link函数的Directive 页面大致是: <b ...
随机推荐
- discuz 门户页模板中的keywords和description不能正常显示
最近用discuz搭建了一个素食网,在处理门户页模板时,发现虽然在后台的seo设置了keywords和description,但是以游客的身份访问时,不显示后台设置的内容,显示为: <meta ...
- 开源框架Caliburn.Micro
Caliburn.Micro学习笔记----引导类和命名匹配规则 用了几天时间看了一下开源框架Caliburn.Micro 这是他源码的地址http://caliburnmicro.codeple ...
- 迟到的 WPF 学习 —— 布局
布局是 WPF 很重头的一部分内容,这一部分梳理和记录关于布局章节的知识点. 1. WPF 使用一种基于流(Flow-based)的概念来处理布局逻辑,将传统的基于"坐标"的思想尽 ...
- 业务接口+UI层的设计(基于Castle实现的Repository)
业务接口+UI层的设计(基于Castle实现的Repository) Repository层设计的文章见:[http://www.cnblogs.com/yomho/p/3297042.html] ...
- Django 中的 WSGI
Django 源码小剖: Django 中的 WSGI 2013-09-06 22:31 by 捣乱小子, 334 阅读, 0 评论, 收藏, 编辑 Django 其内部已经自带了一个方便本地测试的小 ...
- Rustlang语言逐行处理文件的基本方法
文件操作 需求: 将文件中的内容按行读取出来,然后对改行的数据进行处理,最后将处理后的行数据存放到新的文件中. 使用RUST来处理的方法如下. 首先引入需要的标准库: use std::io::pre ...
- nginx 使用过程中一些基础性问题总结
最近闲着无事,玩了下nginx.但本人在实践的过程中也遇到了一些问题,如,大家都知道应用服务器的处理都是无状态的,而nginx做了请求分发,我们在当前web服务器做得提交操作,可能下一刻就跑到另外一台 ...
- COFF/PE文件结构
COFF/PE文件结构 原创 C++应用程序在Windows下的编译.链接(二)COFF/PE文件结构 2.1概述 在windows操作系统下,可执行文件的存储格式是PE格式:在Linux操作系统下, ...
- 对dump脱壳的一点思考
对dump脱壳的一点思考 偶然翻了一下手机日历,原来今天是夏至啊,时间过的真快.ISCC的比赛已经持续了2个多月了,我也跟着比赛的那些题目学了2个月.......虽然过程很辛苦,但感觉还是很幸运的,能 ...
- 对象转Json序列化
C#--对象转Json序列化 前言 最近在研究Android项目,其中涉及到Android中解析Json数据的问题,需要模拟网络中传递Json数据的模式,因为以前是.net的工程师,所以想着从.net ...