4

我创建了这个示例 repo,它使用 rails (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

找到了解决方案:我们需要两个版本的组件MountUp:一个使用浏览器历史记录的客户端版本和一个使用虚假内存历史记录的服务器版本。这里有两个版本的组件:

// 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 回答
1

我认为告诉它不要预渲染无济于事

prerender: false

另外,按照它的建议去做并获取开发版本,这样它会告诉你更多信息

use the non-minified dev environment for the full error message

您告诉它基于History对象构建路由,该对象应该说明用户请求的位置(url)。在服务器端,您将需要以某种方式构造一个对象,该对象将通过一个条目模拟浏览器历史记录:请求的 url。

我已经在使用 Redux 的节点中使用了这个:

import createHistory from 'history/lib/createMemoryHistory';


var reducer = combineReducers(reducers);

const store = compose(
  applyMiddleware(promiseMiddleware, thunk),
  reduxReactRouter({routes, createHistory})
)(createStore)(reducer, {});

但是您需要采取不同的方法才能将 Railsrequest详细信息放入历史存储中。

来自开发库的更好的错误消息是您需要的下一件事。

于 2016-03-11T12:41:43.493 回答