4

レール (v4.2.6) と react-rails (v1.6.2) および react-router (v2.0.0-rc5) を使用するこのサンプル リポジトリを作成しました: https://github.com/pioz/rails_with_react_and_react_router_example

app/views/application/react_entry_point.html.erbコンポーネントをレンダリングするファイルMountUp

<%= react_component('MountUp', {}, {prerender: false}) %>

コンポーネントMountUpは私のルーターをレンダリングします:

class MountUp extends React.Component {
  render() {
    return(
      <Router history={History}>
        <Route path="/" component={App}>
          <IndexRoute component={Index} />
          <Route path="/contact" component={Contact}/>
          <Route path="/about" component={About}/>
        </Route>
      </Router>
    )
  }
}

すべて正常に動作しますが、オプションprerender: trueを変更すると奇妙なエラーが発生しますReact::ServerRendering::PrerenderError in Application#react_entry_point:

Encountered error "Error: Minified exception occurred; use the non-minified dev environment for the full error message and additional helpful warnings." when prerendering MountUp with {}
Object.invariant [as default] ((execjs):21983:16)
createHashHistory ((execjs):24108:130)
(execjs):22633:20
wrapDeprecatedHistory ((execjs):25285:61)
createRouterObjects ((execjs):25259:23)
componentWillMount ((execjs):25228:38)
ReactCompositeComponentMixin.mountComponent ((execjs):8138:13)
wrapper [as mountComponent] ((execjs):3131:22)
Object.ReactReconciler.mountComponent ((execjs):6583:36)
ReactCompositeComponentMixin.mountComponent ((execjs):8153:35)
/Users/pioz/.rvm/gems/ruby-2.3.0/gems/execjs-2.6.0/lib/execjs/external_runtime.rb:39:in `exec'
...

このアプリのサーバー側をレンダリングするにはどうすればよいですか? これはこれを行う正しい方法ですか?

4

2 に答える 2

2

解決策が見つかりました: 2 つのバージョンのコンポーネントが必要ですMountUp: ブラウザの履歴を使用するクライアント バージョンと、偽のメモリ履歴を使用するサーバー バージョンです。コンポーネントの 2 つのバージョンは次のとおりです。

// client version
class MountUp extends React.Component {
  render() {
    return(
      <Router history={History}>
        <Route path="/" component={App}>
          <IndexRoute component={Index} />
          <Route path="/contact" component={Contact}/>
          <Route path="/about" component={About}/>
        </Route>
      </Router>
    )
  }
}


// server version
class MountUp extends React.Component {
  render() {
    return(
      <Router history={createMemoryHistory(this.props.path)}>
        <Route path="/" component={App}>
          <IndexRoute component={Index} />
          <Route path="/contact" component={Contact}/>
          <Route path="/about" component={About}/>
        </Route>
      </Router>
    )
  }
}

また、リクエストと同じ URL パスを持つメモリ履歴を作成する必要があります。これを行うにはpath、リクエストのパスを持つ新しい prop をコンポーネントに渡すことができます。

<%= react_component('MountUp', {path: request.path}, {prerender: true}) %>
于 2016-03-16T16:48:10.497 に答える