(譯)React-Router4的變化

首先,這篇文章的目的並非爲了從新敘述一遍React-Router4的文檔。接下來我要說的內容,將會覆蓋React-Router的大多數API,可是真正的目的是揭開React-Router4成功的模式和策略。css

在開始本文以前,你須要瞭解一些JS的概念:react

  • React無狀態函數式組件
  • ES6箭頭函數以及它的「隱式返回」
  • ES6解構賦值
  • ES6模板字符串

若是你喜歡直接看demo來了解,請點擊此連接查看。web

新的API以及新的心智模型

React-Router早一些的版本中,是在一個地方統一地管理全部路由規則,將它們與佈局組件分離。固然,路由也能夠被拆分和組織到幾個文件當中,可是從概念上來講,路由自己是一個單元,基本上是一個美化的配置文件。npm

要了解版本4到底有哪些不一樣,最好的方式多是用每個版本寫一個兩個頁面的應用,並進行比較。示例的應用只有home頁面和user頁面。redux

如下是版本3的寫法:api

import { Router, Route, IndexRoute } from 'react-router'

const PrimaryLayout = props => (
  <div className="primary-layout">
    <header>
      Our React Router 3 App
    </header>
    <main>
      {props.children}
    </main>
  </div>
)

const HomePage =() => <div>Home Page</div>
const UsersPage = () => <div>Users Page</div>

const App = () => (
  <Router history={browserHistory}>
    <Route path="/" component={PrimaryLayout}>
      <IndexRoute component={HomePage} />
      <Route path="/users" component={UsersPage} />
    </Route>
  </Router>
)

render(<App />, document.getElementById('root'))
複製代碼

版本3中的一些概念在版本4中可能再也不適用:瀏覽器

  • 路由集中在一個文件當中
  • 佈局和頁面嵌套是經過<Route>組件的嵌套派生的
  • 佈局和頁面組件是徹底

React-Router4不在提倡集中管理路由。相反,路由規則存在於佈局和UI組件之中。舉例來講,一樣的路由在V4版本中多是這樣的:bash

import { BrowserRouter, Route } from 'react-router-dom'

const PrimaryLayout = () => (
  <div className="primary-layout">
    <header>
      Our React Router 4 App
    </header>
    <main>
      <Route path="/" exact component={HomePage} />
      <Route path="/users" component={UsersPage} />
    </main>
  </div>
)

const HomePage =() => <div>Home Page</div>
const UsersPage = () => <div>Users Page</div>

const App = () => (
  <BrowserRouter>
    <PrimaryLayout />
  </BrowserRouter>
)

render(<App />, document.getElementById('root'))
複製代碼

新API概念: 因爲咱們的應用是針對瀏覽器的,所以咱們須要將它包裝在V4提供的<BrowserRouter> 組件中。咱們也注意到,使用路由時咱們引入的是react-router-dom,這意味着你在開發時應該安裝react-router-dom 而不是 react-router。這裏有個小小的暗示,既然它的名字是react-router-dom,這意味着它也提供了原生版本。網絡

在使用V4版本構建的應用中,最顯著的一點是「路由」彷佛消失了。在V3中,路由是咱們渲染到dom的一個大玩意兒,它統籌了咱們的整個項目。如今,除了<BrowserRouter> 之外,第一個拋進dom的是咱們的應用自己。react-router

另外一個在V4示例中沒有展示的V3的用法是使用props.children來嵌套組件。這是由於在V4中,只要路由匹配到了,<Router>組件在哪裏編寫,子組件就會渲染在哪裏。

包含式路由

在先前的例子中,你可能已經注意到了exact屬性。那麼它究竟是幹嗎的呢?在V3中,路由規則是獨一無二的,這意味着最終只有一個路由會匹配到。可是在V4中,路由是包含式的,這意味着可能會有多個路由同時匹配到而且渲染。

在先前的例子中,咱們試圖依靠路徑來區分渲染home頁面或者user頁面。若是把exact屬性從該示例中移除,那麼在瀏覽‘/user’路徑時,home頁面和user頁面會同時渲染。

想要更好的瞭解匹配規則,您能夠查看path-to-regexp,React-Router4 就是使用它來匹配路由的。

爲了更好地演示包含式路由起到了什麼做用,咱們實現一個這樣的功能:只有在user頁面咱們纔在頭部添加一個UserMenu列表。

const PrimaryLayout = () => (
  <div className="primary-layout">
    <header>
      Our React Router 4 App
      <Route path="/users" component={UsersMenu} />
    </header>
    <main>
      <Route path="/" exact component={HomePage} />
      <Route path="/users" component={UsersPage} />
    </main>
  </div>
)
複製代碼

如今,當用戶瀏覽‘/users’時,全部的組件都會渲染。在V3中,經過某些方式咱們彷佛也能實現相同的功能,可是實現起來顯然更爲複雜。而V4的包含式組件讓這一切變得輕鬆寫意。

獨立路由

若是你只想匹配一個路由,使用<Switch>組件來實現。<Switch>組件只會渲染匹配到的第一個路由。

const PrimaryLayout = () => (
  <div className="primary-layout">
    <PrimaryHeader />
    <main>
      <Switch>
        <Route path="/" exact component={HomePage} />
        <Route path="/users/add" component={UserAddPage} />
        <Route path="/users" component={UsersPage} />
        <Redirect to="/" />
      </Switch>
    </main>
  </div>
)
複製代碼

<Switch>中的路由只有一個會被渲染。咱們仍然須要在home頁面的路由上添加exact屬性,不然,當咱們訪問‘/users’或者‘/users/add’時,home頁面將會首先匹配到。事實上,在使用排他性路由時,路由的位置排列是最爲關鍵的事,傳統路由一直如此。咱們把‘/users/add’排列在‘/users’以前以保證正確的匹配路由。由於‘/users/add’會匹配‘/users’和‘/users/add’,把‘/users/add’放在前面是最好的。

固然,若是你使用了exact屬性,那放在哪兒都無所謂,但至少咱們還有選擇。

在遇到<Redirect>組件時,老是會作瀏覽器重定向,可是若是它在<Switch>組件中,那麼只有當其餘路由都沒有匹配到時,纔會重定向。

「Index Routes」 和 「Not Found」

在V4中咱們用exact屬性代替了<IndexRoute>來作一樣的事。在沒有路由匹配到時,使用<Redirect>來重定向到默認頁面的路徑,或者404頁面。

嵌套佈局

如今你可能已經開始思考如何實現嵌套佈局了。我本來覺得我不會糾結於這個概念,但我沒有...V4提供了不少選擇,這就是爲何V4如此強大。可是,多樣的選擇意味着咱們可能選擇並不理想的方案。表面上看來,嵌套佈局很簡單,可是根據你的選擇不一樣,你可能會遇到一些麻煩由於你組織路由的方式。

想象這樣一個場景,咱們須要一個‘瀏覽用戶’的頁面以及‘用戶信息’的頁面。對於產品頁面咱們也有相似需求。用戶和產品都須要一個子佈局,而且子佈局都是惟一特有的。舉例來講,每個子佈局擁有不一樣的導航卡。咱們有幾種不一樣的方式來解決這個問題,有些比較好有些則不太好。第一種方案可能並非那麼好,但我仍是想讓大家看一看,以避免之後踩入這個坑。第二種方案就顯得更好一些。

const PrimaryLayout = props => {
  return (
    <div className="primary-layout">
      <PrimaryHeader />
      <main>
        <Switch>
          <Route path="/" exact component={HomePage} />
          <Route path="/users" exact component={BrowseUsersPage} />
          <Route path="/users/:userId" component={UserProfilePage} />
          <Route path="/products" exact component={BrowseProductsPage} />
          <Route path="/products/:productId" component={ProductProfilePage} />
          <Redirect to="/" />
        </Switch>
      </main>
    </div>
  )
}

const BrowseUsersPage = () => (
  <div className="user-sub-layout">
    <aside>
      <UserNav />
    </aside>
    <div className="primary-content">
      <BrowseUserTable />
    </div>
  </div>
)

const UserProfilePage = props => (
  <div className="user-sub-layout">
    <aside>
      <UserNav />
    </aside>
    <div className="primary-content">
      <UserProfile userId={props.match.params.userId} />
    </div>
  </div>
)

複製代碼

新API概念: 任何由<Route>渲染的組件,V4都會提供props.match。就像你看到的,userId這個參數是有props.match.params提供的。瞭解更多V4文檔。若是一個組件並非直接經過<Route>組件渲染的,可是卻要用到props.match,那麼咱們能夠藉助withRouter()這個高階組件。

實現上來講,第一種方案並無什麼問題。可是咱們仔細觀察一下,會發現UserProfilePageBrowserUsersPage的佈局是同樣的,不一樣的只是UserProfileBrowserUserTable。這種實現方法會致使一些冗餘代碼,而且在UserProfilePageBrowserUsersPage之間切換時,UserNav組件是須要徹底從新渲染走一遍生命週期的。很明顯這是能夠避免的。

如下是更好的實現方式:

const PrimaryLayout = props => {
  return (
    <div className="primary-layout">
      <PrimaryHeader />
      <main>
        <Switch>
          <Route path="/" exact component={HomePage} />
          <Route path="/users" component={UserSubLayout} />
          <Route path="/products" component={ProductSubLayout} />
          <Redirect to="/" />
        </Switch>
      </main>
    </div>
  )
}

const UserSubLayout = () => (
  <div className="user-sub-layout">
    <aside>
      <UserNav />
    </aside>
    <div className="primary-content">
      <Switch>
        <Route path="/users" exact component={BrowseUsersPage} />
        <Route path="/users/:userId" component={UserProfilePage} />
      </Switch>
    </div>
  </div>
)
複製代碼

和以前實現方式不一樣的是,咱們的路由從4個變成了2個。BrowseUsersPageUserProfilePage組件只負責渲染不一樣的部分,做爲UserSubLayout的子組件。須要注意的一點是,不論你路由嵌套多深,仍是須要寫完整路徑去匹配。若是你不想重複書寫路徑或者方便統一修改路徑,你可使用props.match.path

const UserSubLayout = props => (
  <div className="user-sub-layout">
    <aside>
      <UserNav />
    </aside>
    <div className="primary-content">
      <Switch>
        <Route path={props.match.path} exact component={BrowseUsersPage} />
        <Route path={`${props.match.path}/:userId`} component={UserProfilePage} />
      </Switch>
    </div>
  </div>
)
複製代碼

Match

正如咱們所見,props.match在獲取userId這類參數和簡化路由書寫方面頗有用處。match對象提供了一些屬性:match.paramsmatch.pathmatch.url等等

match.path 和 match.url

第一眼咱們並不能清楚的區分這二者。控制檯打印時,多數狀況下也是相同的結果。舉例來講,當瀏覽器路徑是‘/users’時,match.pathmatch.url打印輸出的是相同的值。

const UserSubLayout = ({ match }) => {
  console.log(match.url)   // output: "/users"
  console.log(match.path)  // output: "/users"
  return (
    <div className="user-sub-layout">
      <aside>
        <UserNav />
      </aside>
      <div className="primary-content">
        <Switch>
          <Route path={match.path} exact component={BrowseUsersPage} />
          <Route path={`${match.path}/:userId`} component={UserProfilePage} />
        </Switch>
      </div>
    </div>
  )
}
複製代碼

如今咱們仍是分不清這二者的區別,可是,若是咱們用嵌套的路由,特別是帶有參數的匹配模式,在更深一層的組件中打印,咱們就能看到區別。好比咱們瀏覽的是‘/users/5’這個路徑,match.url返回的是‘/users/5’,而match.path返回的是‘/users/:userId’。這樣就一目瞭然了,match.url返回的是具體的路由地址字符串,而match.path返回的是路由匹配模式的字符串。

如何選擇?

在用<Route>作嵌套路由時,建議使用match.path,由於你可能在你的子組件中須要使用match.param對象,在<Link>組件作路由跳轉時,使用match.url

避免match衝突

假設如今咱們須要一個編輯頁面對用戶信息進行編輯,那此頁面的路由規則應該相似於‘/users/:userId/edit’。那麼問題來了,前面的例子中,用戶信息頁面的路由匹配規則是/users/:userId,當咱們訪問‘/users/5/edit’的時候,將會首先匹配到用戶信息頁面而不是編輯頁。那是否意味着咱們要像以前作的那樣,在原有的信息頁面同時添加一個編輯的子頁面,再去匹配呢?並不必定。咱們能夠經過將‘/users/:userId/edit’規則放在/users/:userId以前來達到此效果。或者對/users/:userId進行進一步約定,好比限制:userId爲數字:users/:userId(\\d+)

const UserSubLayout = ({ match }) => (
  <div className="user-sub-layout">
    <aside>
      <UserNav />
    </aside>
    <div className="primary-content">
      <Switch>
        <Route exact path={props.match.path} component={BrowseUsersPage} />
        <Route path={`${match.path}/add`} component={AddUserPage} />
        <Route path={`${match.path}/:userId/edit`} component={EditUserPage} />
        <Route path={`${match.path}/:userId`} component={UserProfilePage} />
      </Switch>
    </div>
  </div>
)
複製代碼

受權路由

在咱們的項目中,根據用戶的登陸狀態來限制用戶訪問某些路由的功能是很常見的。讓受權頁面(應用程序的主要頁面)和未受權頁面(好比登陸頁面以及忘記密碼頁面)擁有不一樣的UI和感覺也是常見的需求。

class App extends React.Component {
  render() {
    return (
      <Provider store={store}>
        <BrowserRouter>
          <Switch>
            <Route path="/auth" component={UnauthorizedLayout} />
            <AuthorizedRoute path="/app" component={PrimaryLayout} />
          </Switch>
        </BrowserRouter>
      </Provider>
    )
  }
}
複製代碼

這種方法有幾個關鍵點。首先,根據咱們所處的應用程序的哪一個部分,我在兩個頂級佈局之間進行選擇。訪問路徑,如「/auth/login」或「/auth/forget-password」將使用未經受權的佈局。當用戶登陸時,咱們將保證全部路徑都有一個' /app '前綴,它使用AuthorizedRoute組件來肯定用戶是否登陸。若是用戶試圖訪問以' /app '開頭的頁面,但他們沒有登陸,他們將被重定向到登陸頁面。

可是AuthorizedRoute並非V4自己提供的,而是我本身實現的。V4中一個驚人的新特性是可以爲特定目的建立本身的路由。不一樣於經過傳遞一個組件屬性到<Route>,而是傳遞一個rendr回調:

class AuthorizedRoute extends React.Component {
  componentWillMount() {
    getLoggedUser()
  }

  render() {
    const { component: Component, pending, logged, ...rest } = this.props
    return (
      <Route {...rest} render={props => {
        if (pending) return <div>Loading...</div>
        return logged
          ? <Component {...this.props} />
          : <Redirect to="/auth/login" />
      }} />
    )
  }
}

const stateToProps = ({ loggedUserState }) => ({
  pending: loggedUserState.pending,
  logged: loggedUserState.logged
})

export default connect(stateToProps)(AuthorizedRoute)
複製代碼

你的登陸策略可能和個人並不同,我用一個網絡請求getLoggedUser()去獲取狀態並把pending和logged的值插入redux管理的狀態中。pending意味着請求還沒結束。

點擊此連接你能夠查看完整的登陸限制的代碼實現。

登陸限制動態圖

其餘注意點

React-Router V4還有其餘更多炫酷的特性。最後,咱們來了解一些小特性,以防遇到時犯迷糊。

<Link><NavLink>

在V4中,有兩種方式能夠將錨點和路由集成:<Link><NavLink>

<NavLink>的原理和<Link>同樣,不一樣的是,<NavLink>能根據瀏覽器URL是否匹配從而提供額外的樣式定製。詳情能夠在線查看該demo。部分代碼以下:

const PrimaryHeader = () => (
  <header className="primary-header">
    <h1>Welcome to our app!</h1>
    <nav>
      <NavLink to="/app" exact activeClassName="active">Home</NavLink>
      <NavLink to="/app/users" activeClassName="active">Users</NavLink>
      <NavLink to="/app/products" activeClassName="active">Products</NavLink>
    </nav>
  </header>
)
複製代碼

<NavLink>匹配到當前URL時,它容許咱們在此<NavLink>上添加一個自定義的class,以便控制樣式。

URL查詢字符串

在V4中已經沒有方式能夠直接獲得URL的查詢字符串了。在我看來,作出這個決定是由於對於如何處理複雜的查詢字符串沒有標準。所以,v4沒有在模塊中引入相關方法,而是決定讓開發人員選擇如何處理查詢字符串。這是一件好事。

就我我的而言,我使用sindresorhus大神(twitter)編寫的query-string模塊來處理查詢字符串。

動態路由

V4最出色的一點是幾乎全部的東西都是一個React組件,包括<Route>。路由不再是什麼魔法,咱們能夠在任何須要的時候有條件的渲染它們。想象一下,當知足某些條件時,您的應用程序的整個部分均可用於路由。當這些條件不知足時,咱們能夠刪除路由。咱們甚至能夠作一些炫酷的事,好比遞歸路由。

React-Router4變得更爲簡單了,由於萬物皆組件。

相關文章
相關標籤/搜索