1 angular.module('uiRouterSample.contacts', [ 2 'ui.router' 3 ]) 4 5 .config( 6 [ '$stateProvider', '$urlRouterProvider', 7 function ($stateProvider, $urlRouterProvider) { 8 $stateProvider 9 // 10 // Contacts // 11 // 12 .state('contacts', { 13 14 // With abstract set to true, that means this state can not be explicitly activated. 15 // It can only be implicitly activated by activating one of its children. 16 /** 17 * 18 * 抽象路由模板,该状态永远不会被激活 19 * 只会被其子状态隐式激活 20 * 在本例子中 就是被导航中的ui-sref=contacts.list激活的 21 * */ 22 abstract: true, 23 24 // This abstract state will prepend '/contacts' onto the urls of all its children. 25 /** 26 * 这个抽象状态会预置/contacts到其所有的子状态中 27 * 28 * */ 29 url: '/contacts', 30 31 // Example of loading a template from a file. This is also a top level state, 32 // so this template file will be loaded and then inserted into the ui-view 33 // within index.html. 34 templateUrl: 'app/contacts/contacts.html', 35 36 // Use `resolve` to resolve any asynchronous controller dependencies 37 // *before* the controller is instantiated. In this case, since contacts 38 // returns a promise, the controller will wait until contacts.all() is 39 // resolved before instantiation. Non-promise return values are considered 40 // to be resolved immediately. 41 /** 42 * resolve处理异步数据调用,在控制器实例化之前 43 *在这个例子中,contacts返回一个promise,控制器会等到promise处理完之后再进行初始化 44 *non-promise会立即resolve 45 */ 46 resolve: { 47 contacts: ['contacts', 48 function( contacts){ 49 return contacts.all(); 50 }] 51 }, 52 53 // You can pair a controller to your template. There *must* be a template to pair with. 54 /*** 55 * 为template添加控制器。 56 * 必须与template配对 57 * 58 */ 59 controller: ['$scope', '$state', 'contacts', 'utils', 60 function ( $scope, $state, contacts, utils) { 61 62 // Add a 'contacts' field in this abstract parent's scope, so that all 63 // child state views can access it in their scopes. Please note: scope 64 // inheritance is not due to nesting of states, but rather choosing to 65 // nest the templates of those states. It's normal scope inheritance. 66 $scope.contacts = contacts; 67 68 $scope.goToRandom = function () { 69 var randId = utils.newRandomKey($scope.contacts, "id", $state.params.contactId); 70 71 72 // $state.go() can be used as a high level convenience method 73 // for activating a state programmatically. 74 /** 75 * go()方法是一个高级又便利的方法 76 *用于激活一个动态的状态 77 */ 78 $state.go('contacts.detail', { contactId: randId }); 79 }; 80 }] 81 }) 82 83 ///// 84 // Contacts > List // 85 ///// 86 87 // Using a '.' within a state name declares a child within a parent. 88 // So you have a new state 'list' within the parent 'contacts' state. 89 /** 90 * 在状态中用一个'.'才分割父状态和子状态, 91 * 所有现在创建了一个新的'list'状态位于父状态'contacts'中 92 */ 93 .state('contacts.list', { 94 95 // Using an empty url means that this child state will become active 96 // when its parent's url is navigated to. Urls of child states are 97 // automatically appended to the urls of their parent. So this state's 98 // url is '/contacts' (because '/contacts' + ''). 99 /** 100 * 用空的url表示这个子状态会在其父状态导航到时被激活 101 * 子状态会自动添加 102 * 所以这个状态的url='/contacts'+'' 103 */ 104 url: '', 105 106 // IMPORTANT: Now we have a state that is not a top level state. Its 107 // template will be inserted into the ui-view within this state's 108 // parent's template; so the ui-view within contacts.html. This is the 109 // most important thing to remember about templates. 110 /** 111 * 现在我们有一个非顶级的状态 112 *这个模板会出入到其父状态的ui-view中:也就是contact.html中的ui-view 113 *这是模板中很重要的性质 114 */ 115 templateUrl: 'app/contacts/contacts.list.html' 116 }) 117 118 /// 119 // Contacts > Detail // 120 /// 121 122 // You can have unlimited children within a state. Here is a second child 123 // state within the 'contacts' parent state. 124 /** 125 * 一个状态的子状态是无限的,这是contacts的第二个子状态 126 */ 127 .state('contacts.detail', { 128 129 // Urls can have parameters. They can be specified like :param or {param}. 130 // If {} is used, then you can also specify a regex pattern that the param 131 // must match. The regex is written after a colon (:). Note: Don't use capture 132 // groups in your regex patterns, because the whole regex is wrapped again 133 // behind the scenes. Our pattern below will only match numbers with a length 134 // between 1 and 4. 135 /** 136 * url上可以有参数,定义形式:param 或者 {param} 137 * 如果使用{},也可以定义成一个正则表达式,参数必须匹配正则 138 * 139 */ 140 141 // Since this state is also a child of 'contacts' its url is appended as well. 142 // So its url will end up being '/contacts/{contactId:[0-9]{1,4}}'. When the 143 // url becomes something like '/contacts/42' then this state becomes active 144 // and the $stateParams object becomes { contactId: 42 }. 145 /** 146 * /contacts/42 147 * 这样的url就能访问该状态 148 * 而且$stateParams对象就变成了{contactId:42} 149 * 150 */ 151 url: '/{contactId:[0-9]{1,4}}', 152 153 // If there is more than a single ui-view in the parent template, or you would 154 // like to target a ui-view from even higher up the state tree, you can use the 155 // views object to configure multiple views. Each view can get its own template, 156 // controller, and resolve data. 157 /** 158 * 如果父template不止一个ui-view 159 * 或者想访问更往上的状态(—from state tree) 160 * 可以使用views对象来配置多views,每个view都可以从他自己的template,controller,resolve data中获取 161 */ 162 163 // View names can be relative or absolute. Relative view names do not use an '@' 164 // symbol. They always refer to views within this state's parent template. 165 // Absolute view names use a '@' symbol to distinguish the view and the state. 166 // So 'foo@bar' means the ui-view named 'foo' within the 'bar' state's template. 167 /** 168 * view的名字有相对和绝对两种 169 * 相对的视图名字不需要用@标志 170 * 这些view总会在当前状态的父template中展示 171 * 172 * 绝对名称的view使用@标志来区分状态和视图view 173 * foo@bar bar状态的foo view 174 */ 175 views: { 176 177 // So this one is targeting the unnamed view within the parent state's template. 178 /** 179 * ''指向父template中未命名的view 180 */ 181 '': { 182 templateUrl: 'app/contacts/contacts.detail.html', 183 controller: ['$scope', '$stateParams', 'utils', 184 function ( $scope, $stateParams, utils) { 185 $scope.contact = utils.findById($scope.contacts, $stateParams.contactId); 186 }] 187 }, 188 189 // This one is targeting the ui-view="hint" within the unnamed root, aka index.html. 190 // This shows off how you could populate *any* view within *any* ancestor state. 191 /** 192 * ui-view=hint 操作的是root 也就是index.html 193 * 194 */ 195 'hint@': { 196 template: 'This is contacts.detail populating the "hint" ui-view' 197 }, 198 199 // This one is targeting the ui-view="menuTip" within the parent state's template. 200 /** 201 * 这个指向父状态template中的menuTip 202 */ 203 'menuTip': { 204 // templateProvider is the final method for supplying a template. 205 // There is: template, templateUrl, and templateProvider. 206 /** 207 * templateProvider是提供模板的底层方法 208 * 209 */ 210 templateProvider: ['$stateParams', 211 function ( $stateParams) { 212 // This is just to demonstrate that $stateParams injection works for templateProvider. 213 // $stateParams are the parameters for the new state we're transitioning to, even 214 // though the global '$stateParams' has not been updated yet. 215 /** 216 *这里仅仅只是展示$stateParams怎么注入templateProvider的 217 */ 218 return '<hr><small class="muted">Contact ID: ' + $stateParams.contactId + '</small>'; 219 }] 220 } 221 } 222 }) 223 224 // 225 // Contacts > Detail > Item // 226 // 227 228 .state('contacts.detail.item', { 229 230 // So following what we've learned, this state's full url will end up being 231 // '/contacts/{contactId}/item/:itemId'. We are using both types of parameters 232 // in the same url, but they behave identically. 233 url: '/item/:itemId', 234 views: { 235 236 // This is targeting the unnamed ui-view within the parent state 'contact.detail' 237 // We wouldn't have to do it this way if we didn't also want to set the 'hint' view below. 238 // We could instead just set templateUrl and controller outside of the view obj. 239 '': { 240 templateUrl: 'app/contacts/contacts.detail.item.html', 241 controller: ['$scope', '$stateParams', '$state', 'utils', 242 function ( $scope, $stateParams, $state, utils) { 243 $scope.item = utils.findById($scope.contact.items, $stateParams.itemId); 244 245 $scope.edit = function () { 246 // Here we show off go's ability to navigate to a relative state. Using '^' to go upwards 247 // and '.' to go down, you can navigate to any relative state (ancestor or descendant). 248 // Here we are going down to the child state 'edit' (full name of 'contacts.detail.item.edit') 249 $state.go('.edit', $stateParams); 250 }; 251 }] 252 }, 253 254 // Here we see we are overriding the template that was set by 'contacts.detail' 255 'hint@': { 256 template: ' This is contacts.detail.item overriding the "hint" ui-view' 257 } 258 } 259 }) 260 261 ///// 262 // Contacts > Detail > Item > Edit // 263 ///// 264 265 // Notice that this state has no 'url'. States do not require a url. You can use them 266 // simply to organize your application into "places" where each "place" can configure 267 // only what it needs. The only way to get to this state is via $state.go (or transitionTo) 268 /** 269 * 当一个状态并不一定非要有url 可以通过$scope.go 或者ui-sref 指向 270 */ 271 .state('contacts.detail.item.edit', { 272 views: { 273 274 // This is targeting the unnamed view within the 'contacts.detail' state 275 // essentially swapping out the template that 'contacts.detail.item' had 276 // inserted with this state's template. 277 '@contacts.detail': { 278 templateUrl: 'app/contacts/contacts.detail.item.edit.html', 279 controller: ['$scope', '$stateParams', '$state', 'utils', 280 function ( $scope, $stateParams, $state, utils) { 281 $scope.item = utils.findById($scope.contact.items, $stateParams.itemId); 282 $scope.done = function () { 283 // Go back up. '^' means up one. '^.^' would be up twice, to the grandparent. 284 /** 285 * 退回父状态 286 * ^.^表示退回两次 即是grandparent 287 */ 288 $state.go('^', $stateParams); 289 }; 290 }] 291 } 292 } 293 }); 294 } 295 ] 296 );
angular.module('uiRouterSample.contacts', [
'ui.router'
])
.config(
[ '$stateProvider', '$urlRouterProvider',
function ($stateProvider, $urlRouterProvider) {
$stateProvider
//
// Contacts //
//
.state('contacts', {
// With abstract set to true, that means this state can not be explicitly activated.
// It can only be implicitly activated by activating one of its children.
/**
*
* 抽象路由模板,该状态永远不会被激活
* 只会被其子状态隐式激活
* 在本例子中 就是被导航中的ui-sref=contacts.list激活的
* */
abstract: true,
// This abstract state will prepend '/contacts' onto the urls of all its children.
/**
* 这个抽象状态会预置/contacts到其所有的子状态中
*
* */
url: '/contacts',
// Example of loading a template from a file. This is also a top level state,
// so this template file will be loaded and then inserted into the ui-view
// within index.html.
templateUrl: 'app/contacts/contacts.html',
// Use `resolve` to resolve any asynchronous controller dependencies
// *before* the controller is instantiated. In this case, since contacts
// returns a promise, the controller will wait until contacts.all() is
// resolved before instantiation. Non-promise return values are considered
// to be resolved immediately.
/**
* resolve处理异步数据调用,在控制器实例化之前
*在这个例子中,contacts返回一个promise,控制器会等到promise处理完之后再进行初始化
*non-promise会立即resolve
*/
resolve: {
contacts: ['contacts',
function( contacts){
return contacts.all();
}]
},
// You can pair a controller to your template. There *must* be a template to pair with.
/***
* 为template添加控制器。
* 必须与template配对
*
*/
controller: ['$scope', '$state', 'contacts', 'utils',
function ( $scope, $state, contacts, utils) {
// Add a 'contacts' field in this abstract parent's scope, so that all
// child state views can access it in their scopes. Please note: scope
// inheritance is not due to nesting of states, but rather choosing to
// nest the templates of those states. It's normal scope inheritance.
$scope.contacts = contacts;
$scope.goToRandom = function () {
var randId = utils.newRandomKey($scope.contacts, "id", $state.params.contactId);
// $state.go() can be used as a high level convenience method
// for activating a state programmatically.
/**
* go()方法是一个高级又便利的方法
*用于激活一个动态的状态
*/
$state.go('contacts.detail', { contactId: randId });
};
}]
})
/
// Contacts > List //
/
// Using a '.' within a state name declares a child within a parent.
// So you have a new state 'list' within the parent 'contacts' state.
/**
* 在状态中用一个'.'才分割父状态和子状态,
* 所有现在创建了一个新的'list'状态位于父状态'contacts'中
*/
.state('contacts.list', {
// Using an empty url means that this child state will become active
// when its parent's url is navigated to. Urls of child states are
// automatically appended to the urls of their parent. So this state's
// url is '/contacts' (because '/contacts' + '').
/**
* 用空的url表示这个子状态会在其父状态导航到时被激活
* 子状态会自动添加
* 所以这个状态的url='/contacts'+''
*/
url: '',
// IMPORTANT: Now we have a state that is not a top level state. Its
// template will be inserted into the ui-view within this state's
// parent's template; so the ui-view within contacts.html. This is the
// most important thing to remember about templates.
/**
* 现在我们有一个非顶级的状态
*这个模板会出入到其父状态的ui-view中:也就是contact.html中的ui-view
*这是模板中很重要的性质
*/
templateUrl: 'app/contacts/contacts.list.html'
})
///
// Contacts > Detail //
///
// You can have unlimited children within a state. Here is a second child
// state within the 'contacts' parent state.
/**
* 一个状态的子状态是无限的,这是contacts的第二个子状态
*/
.state('contacts.detail', {
// Urls can have parameters. They can be specified like :param or {param}.
// If {} is used, then you can also specify a regex pattern that the param
// must match. The regex is written after a colon (:). Note: Don't use capture
// groups in your regex patterns, because the whole regex is wrapped again
// behind the scenes. Our pattern below will only match numbers with a length
// between 1 and 4.
/**
* url上可以有参数,定义形式:param 或者 {param}
* 如果使用{},也可以定义成一个正则表达式,参数必须匹配正则
*
*/
// Since this state is also a child of 'contacts' its url is appended as well.
// So its url will end up being '/contacts/{contactId:[0-9]{1,4}}'. When the
// url becomes something like '/contacts/42' then this state becomes active
// and the $stateParams object becomes { contactId: 42 }.
/**
* /contacts/42
* 这样的url就能访问该状态
* 而且$stateParams对象就变成了{contactId:42}
*
*/
url: '/{contactId:[0-9]{1,4}}',
// If there is more than a single ui-view in the parent template, or you would
// like to target a ui-view from even higher up the state tree, you can use the
// views object to configure multiple views. Each view can get its own template,
// controller, and resolve data.
/**
* 如果父template不止一个ui-view
* 或者想访问更往上的状态(—from state tree)
* 可以使用views对象来配置多views,每个view都可以从他自己的template,controller,resolve data中获取
*/
// View names can be relative or absolute. Relative view names do not use an '@'
// symbol. They always refer to views within this state's parent template.
// Absolute view names use a '@' symbol to distinguish the view and the state.
// So 'foo@bar' means the ui-view named 'foo' within the 'bar' state's template.
/**
* view的名字有相对和绝对两种
* 相对的视图名字不需要用@标志
* 这些view总会在当前状态的父template中展示
*
* 绝对名称的view使用@标志来区分状态和视图view
* foo@bar bar状态的foo view
*/
views: {
// So this one is targeting the unnamed view within the parent state's template.
/**
* ''指向父template中未命名的view
*/
'': {
templateUrl: 'app/contacts/contacts.detail.html',
controller: ['$scope', '$stateParams', 'utils',
function ( $scope, $stateParams, utils) {
$scope.contact = utils.findById($scope.contacts, $stateParams.contactId);
}]
},
// This one is targeting the ui-view="hint" within the unnamed root, aka index.html.
// This shows off how you could populate *any* view within *any* ancestor state.
/**
* ui-view=hint 操作的是root 也就是index.html
*
*/
'hint@': {
template: 'This is contacts.detail populating the "hint" ui-view'
},
// This one is targeting the ui-view="menuTip" within the parent state's template.
/**
* 这个指向父状态template中的menuTip
*/
'menuTip': {
// templateProvider is the final method for supplying a template.
// There is: template, templateUrl, and templateProvider.
/**
* templateProvider是提供模板的底层方法
*
*/
templateProvider: ['$stateParams',
function ( $stateParams) {
// This is just to demonstrate that $stateParams injection works for templateProvider.
// $stateParams are the parameters for the new state we're transitioning to, even
// though the global '$stateParams' has not been updated yet.
/**
*这里仅仅只是展示$stateParams怎么注入templateProvider的
*/
return '<hr><small class="muted">Contact ID: ' + $stateParams.contactId + '</small>';
}]
}
}
})
//
// Contacts > Detail > Item //
//
.state('contacts.detail.item', {
// So following what we've learned, this state's full url will end up being
// '/contacts/{contactId}/item/:itemId'. We are using both types of parameters
// in the same url, but they behave identically.
url: '/item/:itemId',
views: {
// This is targeting the unnamed ui-view within the parent state 'contact.detail'
// We wouldn't have to do it this way if we didn't also want to set the 'hint' view below.
// We could instead just set templateUrl and controller outside of the view obj.
'': {
templateUrl: 'app/contacts/contacts.detail.item.html',
controller: ['$scope', '$stateParams', '$state', 'utils',
function ( $scope, $stateParams, $state, utils) {
$scope.item = utils.findById($scope.contact.items, $stateParams.itemId);
$scope.edit = function () {
// Here we show off go's ability to navigate to a relative state. Using '^' to go upwards
// and '.' to go down, you can navigate to any relative state (ancestor or descendant).
// Here we are going down to the child state 'edit' (full name of 'contacts.detail.item.edit')
$state.go('.edit', $stateParams);
};
}]
},
// Here we see we are overriding the template that was set by 'contacts.detail'
'hint@': {
template: ' This is contacts.detail.item overriding the "hint" ui-view'
}
}
})
/
// Contacts > Detail > Item > Edit //
/
// Notice that this state has no 'url'. States do not require a url. You can use them
// simply to organize your application into "places" where each "place" can configure
// only what it needs. The only way to get to this state is via $state.go (or transitionTo)
/**
* 当一个状态并不一定非要有url 可以通过$scope.go 或者ui-sref 指向
*/
.state('contacts.detail.item.edit', {
views: {
// This is targeting the unnamed view within the 'contacts.detail' state
// essentially swapping out the template that 'contacts.detail.item' had
// inserted with this state's template.
'@contacts.detail': {
templateUrl: 'app/contacts/contacts.detail.item.edit.html',
controller: ['$scope', '$stateParams', '$state', 'utils',
function ( $scope, $stateParams, $state, utils) {
$scope.item = utils.findById($scope.contact.items, $stateParams.itemId);
$scope.done = function () {
// Go back up. '^' means up one. '^.^' would be up twice, to the grandparent.
/**
* 退回父状态
* ^.^表示退回两次 即是grandparent
*/
$state.go('^', $stateParams);
};
}]
}
}
});
}
]
);