angular的uiRouter服务学习(2)

时间:2022-05-09 09:48:02

本篇接着上一篇 angular的uiRouter服务学习(1) 继续讲解uiRouter的用法

本篇主要讲解uiRouter的嵌套状态&嵌套视图

嵌套状态的方法:

状态和状态之间可以互相嵌套,状态的嵌套共有以下几种方式:

1.使用'.state()'进行嵌套. 比如 .state('contact',{}).state('contact.list',{})

2.使用 ui-router.stateHelper 来创建状态嵌套树. 这种方式需要另外引入依赖,所以很少被使用.也就不具体详解了

3.定义状态的'parent'属性,属性值是一个字符串,就是父状态的名字.比如 {parent:'contacts'}

4.定义状态的'parent'属性,属性值是一个对象,对象就是父状态对象.比如 {parent:contacts}

使用.state()来创建嵌套状态:

可以对$stateProvider使用.state()来指定状态的嵌套继承,比如下面的例子:contacts.list就是contacts的一个子状态:

html:

<div>
<a href="contacts">查看视图</a>
<a href="contacts/list">查看嵌套视图</a>
<ui-view>点击链接后内容会被加载在这里</ui-view>
</div>

js

var nest = angular.module('nest',['ui.router']);
nest.config(function($stateProvider){
$stateProvider.state('contacts',{
url:'/contacts',
templateUrl:'contacts.html'
}).state('contacts.list',{
url:'/list',
templateUrl:'contacts.list.html'
})
});

contacts.html

<h4>联系人列表:</h4>
<ui-view></ui-view>

contacts.list.html

<p>我是contacts.list的内容</p>

*需要注意的是,子状态的url是以父状态的url为baseUrl的.所以这里子状态的url应该是'/contacts/list'

定义状态的'parent'属性,属性值是一个字符串:

可以通过状态的parent属性来指定它的父状态.

js:

nest.config(function($stateProvider){
$stateProvider.state('contacts',{
url:'/contacts',
templateUrl:'contacts.html'
}).state('list',{
url:'/list',
templateUrl:'contacts.list.html',
parent:'contacts'
})
});

这种方式和上一中方式的区别在于,子状态不需要使用 '父状态.子状态' 这种格式.但是需要指定子状态的parent属性为'父状态名'

基于对象的状态实现嵌套状态:

如果你不喜欢上面说到的两种方式,那么还可以使用基于对象的状态: 为状态添加一个name属性,然后给子状态设置parent属性为父状态对象.

需要注意的是,使用这种方式,子状态的name属性命名规则必须和.state()方法一样,按照'父状态.子状态'的格式来命名

var contacts = {
name:'parent', //特意不取名叫'contacts'
url:'/contacts',
templateUrl:'contacts.html'
};
var list = {
name:'parent.child',
url:'/list',
templateUrl:'contacts.list.html',
parent:contacts
};
nest.config(function($locationProvider,$stateProvider){
$stateProvider.state(contacts).state(list);
});

状态的注册顺序:

状态的注册顺序无关紧要,你也可以先注册子状态再注册父状态.它会被放在队列里直到父状态被注册的时候才被注册.

父状态必须存在:

如果你只注册了一个子状态,比如contacts.list, 你就必须定义一个叫做contacts的父状态,(定义先后顺序无所谓,但一定要定义),否则的话,将不会有状态被注册.contacts.list这个状态会被放在队列中,直到contacts状态被定义. 如果你不定义父状态,也不会有任何的报错.所以需要注意,为了让子状态生效,必须定义父状态.

状态取名规则:

没有任何两个状态可以拥有相同的名字. 当使用.state()来创建嵌套状态的时候,父状态没有被指定,而是通过推断得到的. 当不使用.state()来创建嵌套状态的时候,parent属性必须被定义,但即使是两个拥有不同父状态的子状态,你依然不能给它们取相同的名字.

状态和视图的嵌套:

当应用处于一个特定的状态时(也就是某个状态被激活时),这个状态的所有祖先状态也都被激活了. 比如上面的所有例子: 当 'contacts.list' 状态被激活,'contacts' 状态也被隐式的激活了,因为它是 'contacts.list'的父状态.

子状态会把它的视图模板加载到父状态的模板的 ui-view 元素中去.

比如上面的这些例子,只要<a href="contacts/list">查看嵌套视图</a>被点击,contacts.list状态被激活,contacts状态也会被隐式的激活.

子状态从父状态继承了什么?

子状态从父状态继承了以下两项:

  • 父状态中通过resolve定义的依赖
  • 父状态自定义的data属性

其余的,比如控制器,模板,url,等都不会被继承...

resolve继承的栗子:

核心代码:

var contacts = {
name:'parent',
url:'/contacts',
templateUrl:'contacts.html',
resolve:{
resA:function(){
return {value:'A'}
}
},
controller:function($scope,resA){
$scope.a = resA.value
}
};
var list = {
name:'parent.child',
url:'/list',
templateUrl:'contacts.list.html',
parent:contacts,
resolve:{
resB:function(resA){
return {value:resA.value+'B'}
}
},
controller:function($scope,resB){
$scope.b = resB.value
}
};
nest.config(function($locationProvider,$stateProvider){
$locationProvider.html5Mode({enabled:true}).hashPrefix('!');
$stateProvider.state(contacts).state(list);
});

contacts.html:

{{a}}    //A

contacts.list.html:

{{a}}    //A
{{b}} //AB

可以看到,contacts.list子状态继承了contacts父状态里resolve的依赖'resA'

需要注意: 如果你想在子状态实例化之前解析完父状态的resolve项里的某个promise依赖,那么,这个promise依赖必须被注入到子状态里去.

data继承的栗子:

核心代码:

var contacts = {
name:'parent',
url:'/contacts',
templateUrl:'contacts.html',
data:{
dataA:'a',
dataB:'b'
},
controller:function($scope,$state){
$scope.a = $state.current.data.dataA;
}
};
var list = {
name:'parent.child',
url:'/list',
templateUrl:'contacts.list.html',
parent:contacts,
controller:function($scope,$state){
$scope.a = $state.current.data.dataA;
$scope.b = $state.current.data.dataB;
}
};
nest.config(function($locationProvider,$stateProvider){
$locationProvider.html5Mode({enabled:true}).hashPrefix('!');
$stateProvider.state(contacts).state(list);
});

contacts.html:

{{a}}    //a

contacts.list.html:

{{a}}    //a
{{b}} //b

可以看到,父状态的data属性都被子状态继承了

状态的继承是基于视图的继承的

注意,scope作用域的继承和状态的继承无关,它只和视图之间的嵌套关系有关.

很有可能有这种情况: 子状态的视图不仅用于被填充到父状态的视图中,也可能被填充到很多其他地方.在这种情况下.只有被填充到父状态的子状态的视图,才能继承父状态的scope,这种继承只是angular本身的scope继承机制,和状态继承机制是无关的.所以被填充到其他地方的视图是不能访问到父状态的scope的.

抽象状态

一个抽象状态它拥有自己的子状态,但是它不能激活自己. 它只能当子状态被激活的时候,隐式的被激活.

如果要让一个状态成为抽象状态,需要给它设置abstract属性为true

下面是几种抽象状态常用的场合:

  • 给子状态们的url提供一个基础url.
  • 插入带有ui-view(s)的视图模板,显示子状态们通用的内容,而ui-view部分供各个子状态不同的视图模板来填充.
    • 可以给视图添加一个控制器
    • 另外,这样也可以把$scope对象继承给子状态.但是注意,就如同上一点所说的,这是通过angular的scope的视图继承机制实现的,不是状态继承    
  • 通过data属性,给子状态们提供数据.
  • 通过resolve属性,给子状态们提供依赖.
  • 通过触发onEnter和onExit回调,来处理一些事情.
  • 以上几种情况的混合.

注意: 抽象状态也需要带有<ui-view>元素来让子状态视图填充,所以,如果你创建抽象状态是为了给子状态提供基础url,提供resolve依赖,data数据,或者调用onEnter和onExit回调,那么你需要在template属性中设置视图模板为:''<ui-view/>".

下面简单举几个栗子:

栗子1:给子状态们的url提供一个基础url.

html:

<div>
<a href="contacts">查看视图</a>
<a href="contacts/list">查看嵌套视图1</a>
<a href="contacts/detail">查看嵌套视图2</a>
<ui-view>点击链接后内容会被加载在这里</ui-view>
</div>

js:

var contacts = {
abstract:true,
name:'parent',
url:'/contacts',
template:'<ui-view/>'
};
var list = {
name:'parent.child',
url:'/list',
templateUrl:'contacts.list.html',
parent:contacts
};
var detail = {
name:'parent.detail',
url:'/detail',
templateUrl:'contacts.detail.html',
parent:contacts
};
nest.config(function($locationProvider,$stateProvider){
$locationProvider.html5Mode({enabled:true}).hashPrefix('!');
$stateProvider.state(contacts).state(list).state(detail);
});

contacts.list.html:

<p>我是contacts.detail</p>

contacts.detail.html:

<p>我是contacts.detail</p>

当url变为'contacts/list'和'contacts/detail'的时候,会激活相应的状态

------------------------------------------------------------------------------------------------------------------------------------------

栗子2: 父状态插入带有ui-view(s)的视图模板,显示子状态们通用的内容,而ui-view部分供各个子状态不同的视图模板来填充.

html同上.

js:

var contacts = {
abstract:true,
name:'parent',
url:'/contacts',
templateUrl:'contacts.html'
};
var list = {
name:'parent.child',
url:'/list',
templateUrl:'contacts.list.html',
parent:contacts
};
var detail = {
name:'parent.detail',
url:'/detail',
templateUrl:'contacts.detail.html',
parent:contacts
};
nest.config(function($locationProvider,$stateProvider){
$locationProvider.html5Mode({enabled:true}).hashPrefix('!');
$stateProvider.state(contacts).state(list).state(detail);
});

contacts.html:

<h4>联系人列表:</h4>
<ui-view></ui-view>

无论切换哪个子状态,contacts.html中的<h4>联系人列表:</h4>部分都是通用的.

------------------------------------------------------------------------------------------------------------------------------------------

栗子3: 混合使用

html:

<div>
<a href="contacts">查看视图</a>
<a href="contacts/list">查看嵌套视图1</a>
<ui-view>点击链接后内容会被加载在这里</ui-view>
</div>

js:

var contacts = {
abstract:true,
name:'parent',
url:'/contacts',
templateUrl:'contacts.html',
controller:function($scope){
$scope.contacts = [{id:0,name:'code_bunny'},{id:1,name:'white_bunny'},{id:2,name:'black_bunny'}]
}
};
var list = {
name:'parent.child',
url:'/list',
templateUrl:'contacts.list.html',
parent:contacts
};
var detail = {
name:'parent.detail',
url:'/detail/:id',
templateUrl:'contacts.detail.html',
parent:contacts,
controller:function($scope,$stateParams){
$scope.id = $stateParams.id
}
};
nest.config(function($locationProvider,$stateProvider){
$locationProvider.html5Mode({enabled:true}).hashPrefix('!');
$stateProvider.state(contacts).state(list).state(detail);
});

contacts.html:

<h4>联系人列表:</h4>
<ui-view></ui-view>

contacts.list.html:

<ul>
<li ng-repeat="contact in contacts">
<a href="contacts/detail/{{contact.id}}">{{contact.name}}</a>
</li>
</ul>

contacts.detail.html:

<p>{{contacts[id]['name']}}</p>

完整代码: https://github.com/OOP-Code-Bunny/angular/tree/master/uiRouter

参考网站: https://github.com/angular-ui/ui-router/wiki/Nested-States-%26-Nested-Views

angular的uiRouter服务学习(2)的更多相关文章

  1. angular的uiRouter服务学习&lpar;4&rpar;

    本篇接着上一篇angular的uiRouter服务学习(3)继续讲解uiRouter的用法 本篇主要讲解uiRouter的url路由 大多数情况下,状态是和url相关联的: 当url改变,激活对应的状 ...

  2. angular的uiRouter服务学习&lpar;3&rpar;

    本篇接着上一篇 angular的uiRouter服务学习(2) 继续讲解uiRouter的用法 本篇主要讲解uiRouter的多个命名的视图 我们可以给ui-view元素添加ui-view的值来给它命 ...

  3. angular的uiRouter服务学习&lpar;1&rpar;

    angular有内置的路由服务$route:angular -- $route API翻译 使用$route可以帮助实现路由的切换,视图的改变,但是这个内置的$route只包含了基本的功能,在很多场合 ...

  4. angular的uiRouter服务学习&lpar;5&rpar; --- &dollar;state&period;includes&lpar;&rpar;方法

    $state.includes方法用于判断当前激活状态是否是指定的状态或者是指定状态的子状态. $state.includes(stateOrName,params,options) $state.i ...

  5. Angular&period;js之服务与自定义服务学习笔记

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...

  6. 成为优秀Angular开发者所需要学习的19件事

    一款to-do app基本等同于前端开发的"Hello world".虽然涵盖了创建应用程序的CRUD方面,但它通常只涉及那些框架或库也能做到的皮毛而已. Angular看起来似乎 ...

  7. angular利用ui-router登录检查

    angular利用ui-router登录检查 SAP都会有这个问题,session过期或者页面被刷新的情况下应该进入登录页. 监听ui-router的satte事件可以实现当state切换的时候检查登 ...

  8. angular的splitter案例学习

    angular的splitter案例学习,都有注释了,作为自己的备忘. <!DOCTYPE html> <html ng-app="APP"> <he ...

  9. 介绍Angular的注入服务

    其实angular的注入服务是挺复杂的,目前看源码也只看懂了一半,为了不误导大家,我也不讲敢讲太复杂,怕自己都理解错了. 首先我们要知道angular的三种注入方式: 第一种:inference va ...

随机推荐

  1. HTML JavaScripts

    JavaScript JavaScript是一门编程语言,浏览器内置了JavaScript语言的解释器,所以在浏览器上按照JavaScript语言的规则编写相应代码之,浏览器可以解释并做出相应的处理. ...

  2. 如何定位死循环或高CPU使用率&lpar;linux&rpar;

    如何定位死循环或高CPU使用率(linux)  确定是CPU过高 使用top观察是否存在CPU使用率过高现象 找出线程 对CPU使用率过高的进程的所有线程进行排序 ps H -e -o pid,tid ...

  3. win server 2008 r2 sharepoint 域环境安装经历

    环境: 物理机:win7(x64,计算机名字:wyman-pc,ip:192.168.10.102)  / sql server 2008 r2(x64) /VM10 虚拟机:win svr 2008 ...

  4. sql语句创建新登录名和设置权限

    use DBName go --新增用户 exec sp_addlogin '用户名','密码','默认数据库名' --添加登录 exec sp_grantdbaccess N'test' --使其成 ...

  5. python 学习5--matplotlib画图实践

    ### Python的强大很大一部分原因在于,它提供有很多已经写好的,可以现成用的对象 学习参考: http://www.cnblogs.com/vamei/archive/2013/01/30/28 ...

  6. Struts2 常用的常量配置

    在struts2-core-2.1.8.1.jar的org.apache.struts2包下面的default.properties资源文件里可以查到常用的常量配置,这些不用刻意的记住:忘记的时候可以 ...

  7. 【转】第一次使用Android Studio时你应该知道的一切配置

    原文网址:http://www.cnblogs.com/smyhvae/p/4390905.html [声明] 欢迎转载,但请保留文章原始出处→_→ 生命壹号:http://www.cnblogs.c ...

  8. IIS的Unicode漏洞攻击

    IIS有十多种常见漏洞,但利用得最多的莫过于Unicode解析错误漏洞.微软IIS 4.0/5.0在Unicode字符解码的实现中存在一个安全漏洞,用户可以远程通过IIS执行任意命令.当IIS打开文件 ...

  9. 安装jdk后出现bash&colon; &period;&sol;java&colon; &sol;lib&sol;ld-linux&period;so&period;2&colon; bad ELF interpreter&colon; 没有那个文件或目录

    用sudo yum install glibc.i686命令安装好glibc之后问题就解决了

  10. &lbrack;国嵌攻略&rsqb;&lbrack;117&rsqb;&lbrack;LED驱动程序设计&rsqb;

    LED程序设计 1.编写内核模块 2.搭建字符驱动框架 3.实现设备方法 头文件 <linux/io.h> writel() 1.编译/安装驱动 make cp leddev.ko ... ...