AngularJS路由系列(5)-- UI-Router的路由約束、Resolve屬性、路由附加數據、路由進入退出事件

 

本系列探尋AngularJS的路由機制,在WebStorm下開發。主要包括:

UI-Router約束路由參數
UI-Router的Resolve屬性
UI-Router給路由附加數據
UI-Router的onEnter和onExit事件

css

 

AngularJS路由系列包括:

一、AngularJS路由系列(1)--基本路由配置
二、AngularJS路由系列(2)--刷新、查看路由,路由事件和URL格式,獲取路由參數,路由的Resolve
三、AngularJS路由系列(3)-- UI-Router初體驗
四、AngularJS路由系列(4)-- UI-Router的$state服務、路由事件、獲取路由參數
五、AngularJS路由系列(5)-- UI-Router的路由約束、Resolve屬性、路由附加數據、路由進入退出事件
六、AngularJS路由系列(6)-- UI-Router的嵌套Statehtml

 

項目文件結構



node_modules/
public/
.....app/
..........bower_components/
...............toastr/
....................toastr.min.css
....................toastr.min.js
...............jquery/
....................dist/
.........................jquery.min.js
...............angular/
....................angular.min.js
...............angular-ui-router/
....................release/
.........................angular-ui-router.min.js
...............angular-route/
.........................angular-route.min.js
..........controllers/
...............HomeController.js
...............AllSchoolsController.js
...............AllClassroomsController.js
...............AllActivityiesController.js
...............ClassroomController.js
...............ClassroomSummaryController.js
...............ClassroomMessageController.js
..........css/
...............bootstrap.cerulean.min.css
..........filters/
...............activityMonthFilter.js
..........services/
...............dataServices.js
...............notifier.js
..........templates/
...............home.html
...............allSchools.html
...............allClassrooms.html
...............allActivities.html
...............classroom.html
...............classroomDetail.html
...............classroom_parent.html
..........app.js
.....index.html
.....favicon.ico
server/
.....data/
.....routes/
.....views/
.....helpers.js
package.json
server.js

node

UI-Router約束路由參數



■ app.js, 約束參數jquery

 

(function(){
    var app = angular.module('app',['ui.router']);
    
    app.config(['$logProvider', '$stateProvider', function($logProvider, $stateProvider){
        $logProvider.debugEnabled(true);
        
        $stateProvider
            .state('home',{
                url: '/',
                templateUrl: '/app/templates/home.html',
                controller: 'HomeController', //也能夠寫成HomeController as home
                controllerAs: 'home'
            })
            .state('schools',{
                url: '/schools',
                controller: 'AllSchoolController',
                controllerAs: 'schools',
                templateUrl: '/app/templates/allSchools.thml'
            })
            .state('classrooms',{
                url:'/classrooms',
                controller: 'AllClassroomsController',
                controllerAs: 'classrooms',
                templateUrl: '/app/tempates/allClassrooms.html'
            })
            .state('activities', {
                url: '/activities',
                controller: 'AllActivitiesController',
                controllerAs: 'activities',
                templateUrl: '/app/templates/allActivities.html'
            })
            .state('classroom_summary', {
                url: '/classrooms/:id',
                templateUrl: '/app/templates/classroom.html',
                controller: 'ClassroomController',
                controllerAs: 'classroom'
            })
            .state('classroom_detail',{
                url: '/classrooms/{id: [0-9]}/detail/{month}',
                templateUrl: '/app/templates/classroomDetail.html',
                controller: 'ClassroomController',
                controllerAs: 'classroom'
            })
    }]);
    
    app.run(['$rootScope', '$log', function($rootScope, $log){
        $rootScope.$on('$stateChangeSuccess', function(event, toState, toParams, fromState, fromParams){
            $log.debug('successfully changed states') ;
            
            $log.debug('event', event);
            $log.debug('toState', toState);
            $log.debug('toParams', toParams);
            $log.debug('fromState', fromState);
            $log.debug('fromParams', fromParams);
        });
        
        $rootScope.$on('$stateNotFound', function(event, unfoundState, fromState, fromParams){
            $log.error('The request state was not found: ' + unfoundState);
        });
        
        $rootScope.$on('$stateChangeError', function(event, toState, toParams, fromState, fromParams, error){
            $log.error('An error occurred while changing states: ' + error);
            
            $log.debug('event', event);
            $log.debug('toState', toState);
            $log.debug('toParams', toParams);
            $log.debug('fromState', fromState);
            $log.debug('fromParams', fromParams);
        });
    }]);
}());

 

url: '/classrooms/{id: [0-9]}/detail/{month}'這個就對id這個路由參數進行了約束。

■ app.js, 路由參數不符合約束的解決辦法

UI-Router爲咱們提供了$urlRouteProvider服務。json

 

(function(){
    var app = angular.module('app',['ui.router']);
    
    app.config(['$logProvider', '$stateProvider', '$urlRouteProvider', function($logProvider, $stateProvider, $urlRouteProvider){
        $logProvider.debugEnabled(true);
        
        //解決路由異常的辦法在這裏
        $urlRouteProvider.otherwise('/');
        
        $stateProvider
            .state('home',{
                url: '/',
                templateUrl: '/app/templates/home.html',
                controller: 'HomeController', //也能夠寫成HomeController as home
                controllerAs: 'home'
            })
            .state('schools',{
                url: '/schools',
                controller: 'AllSchoolController',
                controllerAs: 'schools',
                templateUrl: '/app/templates/allSchools.thml'
            })
            .state('classrooms',{
                url:'/classrooms',
                controller: 'AllClassroomsController',
                controllerAs: 'classrooms',
                templateUrl: '/app/tempates/allClassrooms.html'
            })
            .state('activities', {
                url: '/activities',
                controller: 'AllActivitiesController',
                controllerAs: 'activities',
                templateUrl: '/app/templates/allActivities.html'
            })
            .state('classroom_summary', {
                url: '/classrooms/:id',
                templateUrl: '/app/templates/classroom.html',
                controller: 'ClassroomController',
                controllerAs: 'classroom'
            })
            .state('classroom_detail',{
                url: '/classrooms/{id: [0-9]}/detail/{month}',
                templateUrl: '/app/templates/classroomDetail.html',
                controller: 'ClassroomController',
                controllerAs: 'classroom'
            })
    }]);
    
    app.run(['$rootScope', '$log', function($rootScope, $log){
        $rootScope.$on('$stateChangeSuccess', function(event, toState, toParams, fromState, fromParams){
            $log.debug('successfully changed states') ;
            
            $log.debug('event', event);
            $log.debug('toState', toState);
            $log.debug('toParams', toParams);
            $log.debug('fromState', fromState);
            $log.debug('fromParams', fromParams);
        });
        
        $rootScope.$on('$stateNotFound', function(event, unfoundState, fromState, fromParams){
            $log.error('The request state was not found: ' + unfoundState);
        });
        
        $rootScope.$on('$stateChangeError', function(event, toState, toParams, fromState, fromParams, error){
            $log.error('An error occurred while changing states: ' + error);
            
            $log.debug('event', event);
            $log.debug('toState', toState);
            $log.debug('toParams', toParams);
            $log.debug('fromState', fromState);
            $log.debug('fromParams', fromParams);
        });
    }]);
}());

 

■ UI-Router的params屬性設置路由參數bootstrap

 

(function(){
    var app = angular.module('app',['ui.router']);
    
    app.config(['$logProvider', '$stateProvider', '$urlRouteProvider', function($logProvider, $stateProvider, $urlRouteProvider){
        $logProvider.debugEnabled(true);
        
        //解決路由異常的辦法在這裏
        $urlRouteProvider.otherwise('/');
        
        $stateProvider
            .state('home',{
                url: '/',
                templateUrl: '/app/templates/home.html',
                controller: 'HomeController', //也能夠寫成HomeController as home
                controllerAs: 'home'
            })
            .state('schools',{
                url: '/schools',
                controller: 'AllSchoolController',
                controllerAs: 'schools',
                templateUrl: '/app/templates/allSchools.thml'
            })
            .state('classrooms',{
                url:'/classrooms',
                controller: 'AllClassroomsController',
                controllerAs: 'classrooms',
                templateUrl: '/app/tempates/allClassrooms.html'
            })
            .state('activities', {
                url: '/activities',
                controller: 'AllActivitiesController',
                controllerAs: 'activities',
                templateUrl: '/app/templates/allActivities.html'
            })
            .state('classroom_summary', {
                url: '/classrooms/:id',
                templateUrl: '/app/templates/classroom.html',
                controller: 'ClassroomController',
                controllerAs: 'classroom'
            })
            .state('classroom_detail',{
                url: '/classrooms/{id: [0-9]}/detail/{month}',
                templateUrl: '/app/templates/classroomDetail.html',
                controller: 'ClassroomController',
                controllerAs: 'classroom',
                params: {
                    classroomMessage: { value: 'Learning is fun!'}
                }
            })
    }]);
    
    app.run(['$rootScope', '$log', function($rootScope, $log){
        $rootScope.$on('$stateChangeSuccess', function(event, toState, toParams, fromState, fromParams){
            $log.debug('successfully changed states') ;
            
            $log.debug('event', event);
            $log.debug('toState', toState);
            $log.debug('toParams', toParams);
            $log.debug('fromState', fromState);
            $log.debug('fromParams', fromParams);
        });
        
        $rootScope.$on('$stateNotFound', function(event, unfoundState, fromState, fromParams){
            $log.error('The request state was not found: ' + unfoundState);
        });
        
        $rootScope.$on('$stateChangeError', function(event, toState, toParams, fromState, fromParams, error){
            $log.error('An error occurred while changing states: ' + error);
            
            $log.debug('event', event);
            $log.debug('toState', toState);
            $log.debug('toParams', toParams);
            $log.debug('fromState', fromState);
            $log.debug('fromParams', fromParams);
        });
    }]);
}());

 

■ ClassroomController.js, 接受更多的路由參數promise

 

(function(){
    angular.module('app')
        .controller('ClassroomController',['dataService', 'notifier', '$stateParams', ClassroomController]);
        
    function ClassroomController(dataService, notifier, $stateParams){
        var vm = this;
        
        vm.month = $stateParams.month;
        
        //接受param設置的參數
        vm.message = $stateParams.classroomMessage;
        
        
        dataService.getClassroom($stateParams.id)
            .then(function(classroom){
                vm.currentClassroom = classroom;
                
                if($stateParams.month){
                    if(classroom.activities.length > 0){
                        vm.timePeriod = dataService.getMonthName($stateParams.month);
                    } else {
                        vm.timePeriod = 'No activities this month';
                    }
                }
                else{
                    vm.timePeriod = 'All activities';
                }
            })
            .catch(showError);
            
        function showError(message){
            notifier.error(message);
        }
        
    }
}());

 

■ classroomDetal.html

{{classroom.message}}



app

UI-Router的Resolve屬性



■ resolve 屬性ide

 

.state('activities',{
    url: '/activities',
    controller: 'AllAcivitiesController',
    controllerAs: 'activities',
    templateUrl: '/app/templates/allActivities.html',
    resolve: {
        activities: function(dataService){
            return dataService.getAllActiviites();
        }
    }
})

 

● 能夠被注入到controller中去
● 返回一個object對象,對象的屬性自定義,屬性值是一個返回promise的函數
● promises必須被resolve,在變化發生以前函數

 

■ app.js, 添加resolve屬性

 

.state('activities',{
    url: '/activities',
    controller: 'AlLActivitiesController',
    controllerAs: 'activities',
    templateUrl: '/app/templates/allActivities.html',
    resolve: {
        activities: function(dataService){
            return dataService.getAllActivites();
        }
    }
})

 

■ AllActivitiesController.js,從resolve中獲取數據

 

(function(){
    angular.module('app')
        .controller('AllActivitiesController',['dataService','notifier','$state','activites', AllActivitiesController]);
        
    function AllActivitiesController(dataService, notifier, $state, activities){
        var vm = this;
        
        vm.selectedMonth = 1;
        vm.allActivities = activities;
        
        vm.search = function(){
            $state.go('classroom_detail',{id:vm.selectedClassroom.id, month: vm.selectedMonth});
        }
        
        dataService.getAllClassrooms()
            .then(function(classroom){
                vm.allClassrooms = classrooms;
                vm.selectedClassroom = classrooms[0];
            })
            .catch(showError);
            
        function showError(message){
            notifier.error(message);
        }
    }
}());

 


當點擊Activites的時候,數據已經在controller,不須要從新獲取,大大減小了頁面加載時間。

UI-Router給路由附加數據



■ 給states附加數據

 

.state('activities',{
    url: '/activities',
    controller: 'AllActivitesController',
    templateUlr: '/app/templates/allActivities.html',
    data: {
        name: 'MyActivity',
        desc: 'Fun!'
    }
})

 

● data中定義的屬性是任意的
● 能被子state繼承

■ app.js, 添加附加數據

 

.state('activities',{
    url: '/activities',
    controller: 'AlLActivitiesController',
    controllerAs: 'activities',
    templateUrl: '/app/templates/allActivities.html',
    resolve: {
        activities: function(dataService){
            return dataService.getAllActivites();
        }
    },
    data: {
        name: 'My Activity',
        desc: 'Func!'
    },
    foo: {
        myFoo: 'bar'
    }
})

 

■ AllActivitiesController.js,從resolve中獲取數據

 

(function(){
    angular.module('app')
        .controller('AllActivitiesController',['dataService','notifier','$state','activites','$log', AllActivitiesController]);
        
    function AllActivitiesController(dataService, notifier, $state, activities, $log){
        var vm = this;
        
        vm.selectedMonth = 1;
        vm.allActivities = activities;
        
        $log.debug($state.current.data);
        $log.debug($state.current.foo);
        
        vm.search = function(){
            $state.go('classroom_detail',{id:vm.selectedClassroom.id, month: vm.selectedMonth});
        }
        
        dataService.getAllClassrooms()
            .then(function(classroom){
                vm.allClassrooms = classrooms;
                vm.selectedClassroom = classrooms[0];
            })
            .catch(showError);
            
        function showError(message){
            notifier.error(message);
        }
    }
}());

 

 

UI-Router的onEnter和onExit事件

 

.state('classroom',{
    url:'/clsssrooms',
    controller:'AllClssroomsController',
    controllerAs: 'classrooms',
    templateUrl: '/app/tempaltes/allClassrooms.html',
    onEnter: function($log){
        $log.debug('Entering the classrooms state');
    },
    onExit: function($log){
        $log.debug('Existing the classrooms state');
    }
})

 

 

 

未完待續~~

相關文章
相關標籤/搜索