AngularJs ui-router 路由的簡單介紹

以前有寫過一篇關於Angular自帶的路由:ngRoute。今天來講說Angular的第三方路由:ui-router。那麼有人就會問:爲何Angular有了自帶的路由,咱們還須要用ui-router呢?這裏簡單明瞭的說明下ngRoute和ui-router的區別吧,其實也沒很大的區別,主要的就是ngRoute針對於單視圖,而ui-router可用於多視圖(這裏說的視圖是指在頁面內咱們可控制的,可變化的區域)。
好比咱們點擊了一個link,咱們須要在視圖中跳轉到指定的一個頁面,那麼ngRoute已經知足了咱們的需求,而當咱們點擊的時候,須要在分別在不一樣的地方跳轉兩個不一樣的頁面的時候,ngRoute就不夠用了,咱們就須要用到ui-router。
這裏咱們仍是先來學習下ui-router(一些簡單的服務和用法)html

ui-router後端

$urlRouterProvider
$urlRouterProvider負責監聽$location.當$location變化的時候,$urlRouterProvider開始在一個規則的列表中一個個的查找,直到找到匹配的值。$urlRouterProvider用於在後端指定url的狀態配置。全部的url被編譯成UrlMatcher對象。
依賴: $urlMatcherFactoryProvider   $locationProvider
方法
deferIntercept(defer);
禁用(或啓用)延遲location變化的攔截。若是你想定製與URL同步的行爲(例如,你須要保持當前的URL去而且推遲一個變化),那麼在配置的時候使用這個方法。
參數
defer:boolean,肯定是禁止仍是啓用該攔截。
代碼:app

  angular.module('Demo',['ui.router'])
  .config(["$urlRouterProvider",function(){
     $urlRouterProvider.deferIntercept(defer); // defer = true/false 
  }])

這是源碼部分:ide

  this.deferIntercept = function (defer) {
    if (defer === undefined) defer = true;
    interceptDeferred = defer; // 默認是true
  };

otherwise(rule);
定義一個當請求的路徑是無效路徑時跳轉的路徑。
rule:你想重定向的url路徑或一個返回的網址路徑的規則函數。函數傳入兩個參數:$injector和$location服務,並且必須返回一個string的url。
代碼:函數

  angular.module('Demo',['ui.router'])
  .config(["$urlRouterProvider",function(){
     $urlRouterProvider.otherwise(rule); // rule = 重定向的url規則
  }])

rule(rule);
定義使用$urlRouterProvider 來匹配指定的URL的規則。
參數
rule:將$injector和$location做爲arguments傳入的處理函數。用來返回一個string類型的url路徑。
代碼:oop

  angular.module('Demo',['ui.router'])
  .config(["$urlRouterProvider",function($urlRouterProvider){
     $urlRouterProvider.rule(function ($injector, $location) {
        var path = $location.path(),
        normalized = path.toLowerCase();
        if (path !== normalized) {
           return normalized;
        }
    });
  }])

when(what,handler);
爲給定的URL匹配註冊一個處理程序。
參數
what:須要重定向的傳入路徑。
handler:你想要重定向的路徑/處理程序。
代碼:post

  angular.module('Demo', ['ui.router']);
  .config(["$urlRouterProvider",function ($urlRouterProvider) {
    $urlRouterProvider.when($state.url, function ($match, $stateParams) {
      if ($state.$current.navigable !== state || !equalForKeys($match, $stateParams) {
       $state.transitionTo(state, $match, false);
      }
    });
  }]);

 

$urlRouter
依賴:$location $rootScope $injector $browser
方法
href(urlMacther,params,options);
一個生成URL的方法,爲給定的UrlMatcher返回編譯後的URL,而且用提供的參數填充。
參數
urlMacther:用於看成生成URL的模板的UrlMacther對象。
params:一個參數值的對象用來填補所需的匹配參數。
options:option對象,absolute-boolean,若是爲true,將會生成一個絕對地址。
代碼:學習

  $bob = $urlRouter.href(new UrlMatcher("/about/:person"), {
    person: "bob"
  });
  // $bob == "/about/bob";
  sync();

sync();
觸發更新:發生在地址欄URL變化時執行相同的更新。ui

 

$state
$state服務負責表明狀態及提供狀態之間的轉換。它還提供你當前的狀態及上一個狀態。
依賴:$rootScope $q $view $injector $resolve $stateParams $urlRouter
方法
get(stateOrName,context);
返回任何指定的狀態或全部狀態的配置對象。
參數
stateOrName:若是提供此參數,返回的是指定狀態的配置對象;若是不提供此參數,則返回所有狀態的配置對象。
context:當context是一個相對的參考狀態,狀態會在相關上下文中檢索。this

方法

go(to,params,options);
參數
to:string,即將跳轉的狀態。
params:object,跳轉所帶的參數。
options:object,可選配置對象。有 location(是否更新地址欄的url,或以什麼字符串替換url),inherit(是否繼承當前url的參數),relative(當變化相對路徑:如"^,定義的狀態是相對的),notify(是否廣播$stateChangeStart和$stateChangeSuccess事件),reload(是否從新載入)。
代碼:

  $state.go('contact.detail');

href(stateOeName,params,options);
一個URL生成方法,返回爲給定的狀態填充指定的參數編譯後的連接。
參數
stateOeName:string,你想要生成的url的狀態或者狀態對象。
params:object,一個用於填充狀態須要的參數的對象。
options:可選配置對象。有lossy(當第一個參數url未被提供時是否繼承導航的url進行構建href),inherit(是否繼承當前url的參數),relative(當變化相對路徑:如"^,定義的狀態是相對的),absolute(是否生成絕對url)。
代碼:

  $state.href("about.person", { person: "bob" })

include(stateOrName,params,options);
一個肯定當前有效的狀態是否是與stateOrName平級的仍是其子狀態。
參數
stateOeName:string,部分名稱,相對名稱,或者經過當前狀態進行全局模式查找。
params:object,一個參數對象。
options:可選配置對象。有relative。
代碼:

  <div ng-class="{highlighted:$state.includes('.item')}">Item</div>
  $state.$current.name = 'contacts.details.item';
  $state.includes("contacts"); // true
  $state.includes("contacts.details"); // true
  $state.includes("contacts.details.item"); // true
  $state.includes("contacts.list"); // false
  $state.includes("about"); // false

全局模式:

  $state.$current.name = 'contacts.details.item.url';
  $state.includes("*.details.*.*"); // true
  $state.includes("*.details.**"); // true
  $state.includes("**.item.**"); // true
  $state.includes("*.details.item.url"); // true
  $state.includes("*.details.*.url"); // true
  $state.includes("*.details.*"); // false  
  $state.includes("item.**"); // false

is(stateOrName,params,options);
與$state.include類似,只是這個針對的是全名。參數性質同上。
代碼:

  <div ng-class="{highlighted: $state.is('.item')}">Item</div>
  $state.$current.name = 'contacts.details.item';
  $state.is('contact.details.item'); // true
  $state.is(contactDetailItemStateObject); // true

reload(state);
從新載入當前狀態的方法。
參數
state:一個狀態名稱或者狀態對象。
代碼:

  $state.reload('contact.detail');

transitionTo(to,toParams,options);
過渡到一個新狀態的方法。
參數
to:狀態名稱。
toParams:將會發送到下一個狀態的參數。
options:可選參數。有location,inherit,relative,notify,reload。
代碼:

  $state.transitionTo($state.current, $stateParams, { 
      reload: true, inherit: false, notify: true
  });

事件
$stateChangeError
路由狀態變化發生錯誤時觸發的事件。參數有:event,toState,toParams,fromState,fromParams,error。以上根據字面意思便可理解,哈哈。
$stateChangeStart
路由狀態變化發生前觸發的事件。參數有:event,toState,toParams,fromState,fromParams。
$stateChangeSuccess
路由狀態變化正確時觸發的事件。參數有:event,toState,toParams,fromState,fromParams。
$stateNotFound
路由狀態沒找到的時候觸發的事件。參數有:event,unfoundState,fromState,fromParams。

 

$stateProvider
處理路由狀態的服務,路由的狀態反映了該項在應用程序中的位置,描述了在當前狀態下UI是應該怎麼樣的,而且該作什麼。

依賴:$urlRouterProvider $urlMatcherFactoryProvider
方法
decorator(name,func);
經過內部的$stateProvider以擴展或者重寫狀態生成器。可用於添加ui-router的自定義功能,例如,基於狀態名稱推斷templateUrl。
警告:由於生成器的函數執行順序的不肯定,decorator不該該相互依賴。
參數:
name:須要修改的生成函數名稱。
func:負責修改生成器函數的函數。
代碼:

  $stateProvider.decorator('views', function (state, parent) {
    var result = {},
        views = parent(state);
    angular.forEach(views, function (config, name) {
      var autoName = (state.name + '.' + name).replace('.', '/');
      config.templateUrl = config.templateUrl || '/partials/' + autoName + '.html';
      result[name] = config;
    });
    return result;
  });
  $stateProvider.state('home', {
    views: {
      'contact.list': { controller: 'ListController' },
      'contact.item': { controller: 'ItemController' }
    }
  });
  $state.go('home');

以上代碼修飾了「views」直接經過state的名稱綁定完對應的頁面模板。
state(name,stateConfig);
註冊一個狀態,並給定其配置。
參數
name:狀態的名稱。
stateConfig:狀態配置對象。配置具備如下各項屬性:
template: string/function,html模板字符串,或者一個返回html模板字符串的函數。
templateUrl:string/function,模板路徑的字符串,或者返回模板路徑字符串的函數。
templateProvider:function,返回html模板字符串或模板路徑的服務。
controller:string/function,新註冊一個控制器函數或者一個已註冊的控制器的名稱字符串。
controllerProvider:function,返回控制器或者控制器名稱的服務
controllerAs:string,控制器別名。
parent:string/object,手動指定該狀態的父級。
resolve:object,將會被注入controller去執行的函數,<string,function>形式。
url:string,當前狀態的對應url。
views:object,視圖展現的配置。<string,object>形式。
abstract:boolean,一個永遠不會被激活的抽象的狀態,但能夠給其子級提供特性的繼承。默認是true。
onEnter:function,當進入一個狀態後的回調函數。
onExit:function,當退出一個狀態後的回調函數。
reloadOnSearch:boolean,若是爲false,那麼當一個search/query參數改變時不會觸發相同的狀態,用於當你修改$location.search()的時候不想從新加載頁面。默認爲true。
data:object,任意對象數據,用於自定義配置。繼承父級狀態的data屬性。換句話說,經過原型繼承能夠達到添加一個data數據從而整個樹結構都能獲取到。
params:url裏的參數值,經過它能夠實現頁面間的參數傳遞。


ui-sref
一種將連接(<a>標籤)綁定到一個狀態的指令。點擊該連接將觸發一個能夠帶有可選參數的狀態轉換。
代碼:

  <a ui-sref="app.index">首頁</a>
  <!-- 這裏是正常的跳轉 -->
  <a ui-sref="app.index({id:yourId})">你的主頁</a>
  <!-- 這裏是帶參數對象的跳轉,名稱是id,值是yourId -->

簡單的使用代碼(ui-router的單視圖):

  <div ng-app="Demo" ng-controller="testCtrl as ctrl">
      <ol>
      <li><a ui-sref="app">app</a></li>
      <li><a ui-sref="test">test</a></li>
      </ol>
        <div ui-view></div>
      <script type="text/ng-template" id="'page1.html'">
          this is page 1 for app.
      </script>
      <script type="text/ng-template" id="'page3.html'">
          this is page 1 for test.
      </script>
  </div>
  angular.module('Demo', ['ui.router'])
  .config(["$stateProvider","$urlRouterProvider",routeConfig])
  .controller("testCtrl", angular.noop)
  function routeConfig($stateProvider,$urlRouterProvider){
      $urlRouterProvider.otherwise("/app");
      $stateProvider
      .state("app",{
          url:"/app",
          templateUrl:"'page1.html'"
      })
      .state("test",{
          url:"/test",
          templateUrl:"'page3.html'"
      })
  }

使用代碼(ui-router的多視圖):

  <div ng-app="Demo" ng-controller="testCtrl as ctrl">
      <ol>
      <li><a ui-sref="app.page1">app</a></li>
      <li><a ui-sref="test.page1({id:1})">test</a></li>
      </ol>
      <div ui-view></div>
      <script type="text/ng-template" id="'layout.html'">
          <div ui-view="nav@"></div>
          <div ui-view></div>
      </script>
      <script type="text/ng-template" id="'nav1.html'">
          <ol>
          <li><a ui-sref="app.page1">app.page1</a></li>
          <li><a ui-sref="app.page2">app.page2</a></li
          </ol>
      </script>
      <script type="text/ng-template" id="'nav2.html'">
          <ol>
          <li><a ui-sref="test.page1({id:1})">test.page1</a></li>
          <li><a ui-sref="test.page2">test.page2</a></li
          </ol>
      </script>
      <script type="text/ng-template" id="'page1.html'">
      this is page 1 for app.
      </script>
      <script type="text/ng-template" id="'page2.html'">
      this is page 2 for app.
      </script>
      <script type="text/ng-template" id="'page3.html'">
      this is page 1 for test.
      </script>
      <script type="text/ng-template" id="'page4.html'">
      this is page 2 for test.
      </script>
  </div>
  angular.module('Demo', ['ui.router'])
  .config(["$stateProvider","$urlRouterProvider",routeConfig])
  .controller("testCtrl", angular.noop)
  function routeConfig($stateProvider,$urlRouterProvider){
      $urlRouterProvider.otherwise("/app/page1");
      $stateProvider
      .state("app",{
            url:"/app",
            views:{
                "":{
                    templateUrl:"'layout.html'"
                },
                "nav":{
                    templateUrl:"'nav1.html'"
                }
            }
        })
        .state("app.page1",{
            url:"/page1",
            templateUrl:"'page1.html'"
        })
        .state("app.page2",{
            url:"/page2",
            templateUrl:"'page2.html'"
        })
        .state("test",{
            url:"/test",
            views:{
            "":{
                templateUrl:"'layout.html'"
            },
             "nav":{
                    templateUrl:"'nav2.html'"
                }   
        }
        })
        .state("test.page1",{
            url:"/page1?:id",
            templateUrl:"'page3.html'",
            controller:["$stateParams",function($stateParams){
                console.log($stateParams.id);// 1  這裏實現傳參
            }],
            params:{
                id:null
            }
        })
        .state("test.page2",{
            url:"/page2",
            templateUrl:"'page4.html'"
        })
  }

注意:須要引入angular-ui-router[.min].js

這裏本獸沒有也不會把所有用法寫一遍 -。- 寫兩個簡單的案例僅供學習參考。這裏偷個懶,把代碼都寫在一個頁面上完成了...   畢竟花了兩個晚上的空閒時間用來整理內容和寫demo  考慮到次日要上班,都是儘可能的早睡,因此整理資料拖的久了些。

相關文章:AngularJs 動態加載模塊和依賴   Angular 學習 -- ngRoute Angular自帶的路由

相關文章
相關標籤/搜索