2017-06-21 65 views
6

我是新來的graphql與react-appollo我想使用reax apollo與redux也服務器端渲染每一件事情都很好我的應用程序工作,但問題是當我的應用程序渲染它實際上是再次調用api它不是用我的再現狀態..如何避免客戶端使用redux重新獲取react-apollo SSR?

enter image description here

服務器.js文件

import express from 'express'; 
import bodyParser from 'body-parser'; 

import path from 'path'; 
import expressGraphQL from 'express-graphql'; 
import schema from './GraphQL/Schema'; 
import React from 'react'; 
import ReactDOMServer from 'react-dom/server' 
import { StaticRouter } from 'react-router'; 
import { ApolloClient, createNetworkInterface, ApolloProvider } from 'react-apollo'; 
import { getDataFromTree } from "react-apollo" 
import store from '../client/Redux/Store/store'; 

import {serverClient} from './lib/apollo' 

require('es6-promise').polyfill(); 
require('isomorphic-fetch'); 

import WApp from '../client/App'; 

//Dev HMR 
import HMR from './serverUtils/HMR'; 

const app = express(); 
app.use(bodyParser.json()); 

app.use('/api', expressGraphQL({ 
    schema, 
    graphiql: true 
})); 
app.use('/static',express.static('build')); 
HMR(app); 

function Html({ content, state }) { 
    return (
     <html> 
     <body> 
     <div id="app" dangerouslySetInnerHTML={{ __html: content }}/> 
     <script src="/static/app.js" /> 
     <script dangerouslySetInnerHTML={{ 
      __html: `window.__APOLLO_STATE__=${JSON.stringify(state).replace(/</g, '\\u003c')};`, 
     }} /> 
     </body> 
     </html> 
    ); 
} 

function createReactHandler(req) { 
    return async function reactHandler(ctx) { 
     const routeContext = {}; 
     const client = serverClient(); 

     const components = (
      <StaticRouter location={req.url} context={routeContext}> 
       <ApolloProvider store={store} client={client}> 
        <WApp /> 
       </ApolloProvider> 
      </StaticRouter> 
     ); 

     await getDataFromTree(components); 

     // const html = ReactDOMServer.renderToString(components); 

     // // Handle redirects 
     // if ([301, 302].includes(routeContext.status)) { 
     //  // 301 = permanent redirect, 302 = temporary 
     //  ctx.status = routeContext.status; 
     // 
     //  // Issue the new `Location:` header 
     //  ctx.redirect(routeContext.url); 
     // 
     //  // Return early -- no need to set a response body 
     //  return; 
     // } 
     // 
     // // Handle 404 Not Found 
     // if (routeContext.status === 404) { 
     //  // By default, just set the status code to 404. You can add your 
     //  // own custom logic here, if you want to redirect to a permanent 
     //  // 404 route or set a different response on `ctx.body` 
     //  ctx.status = routeContext.status; 
     // } 

     // return html; 
     // console.log(html) 


    } 
} 



const HTML = ({ html,state}) => (

    <html lang="en" prefix="og: http://ogp.me/ns#"> 
    <head> 
     <meta charSet="utf-8" /> 
     <meta httpEquiv="X-UA-Compatible" content="IE=edge" /> 
     <meta httpEquiv="Content-Language" content="en" /> 
     <meta name="viewport" content="width=device-width, initial-scale=1" /> 

    </head> 
    <body> 
    <div 
     id="app" 
     dangerouslySetInnerHTML={{ __html: html }} /> 
    <script dangerouslySetInnerHTML={{ 
     __html: `window.__STATE__=${JSON.stringify(state)};`, 
    }} /> 

    <script src="/static/app.js" /> 

    </body> 
    </html> 
); 

app.get('/*',(req,res) => { 
    const routeContext = {}; 
    const client = serverClient(); 

    const components = (
     <StaticRouter location={req.url} context={routeContext}> 
      <ApolloProvider store={store} client={client}> 
       <WApp /> 
      </ApolloProvider> 
     </StaticRouter> 
    ); 

    getDataFromTree(components).then(() => { 
     const html = ReactDOMServer.renderToString(components); 
     const initialState = {apollo: client.getInitialState()} 

     console.log(client); 

     res.send(`<!DOCTYPE html>\n${ReactDOMServer.renderToStaticMarkup(
      <HTML 
       html={html} 
       state={initialState} 
       />, 
     )}`) 
    }) 
}) 




app.listen(3000,() => { 
    console.log('Man I on') 
}) 

store.js

import { createStore, compose, applyMiddleware } from 'redux'; 
import { syncHistoryWithStore } from 'react-router-redux'; 
import thunk from 'redux-thunk'; 
import {createLogger} from 'redux-logger'; 


import client from '../apolloClient'; 
import rootReducer from '../Reducers' 

//All Reducer 
import {initialState as allPosts} from '../Reducers/AllPosts_Reucer'; 
const isProduction = process.env.NODE_ENV !== 'development'; 
const isClient = typeof document !== 'undefined'; 
const initialState = { 
    allPosts 
}; 

const middlewares = [thunk, client.middleware()]; 
const enhancers = []; 

if (!isProduction && isClient) { 
    const loggerMiddleware = createLogger(); 
    middlewares.push(loggerMiddleware); 

    if (typeof devToolsExtension === 'function') { 
     const devToolsExtension = window.devToolsExtension; 
     enhancers.push(devToolsExtension()); 
    } 
} 


const composedEnhancers = compose(
    applyMiddleware(...middlewares), 
    ...enhancers 
); 
const store = createStore(
    rootReducer, 
    {}, 

    composedEnhancers, 
); 

export default store; 

apolloClient.js

import ApolloClient, { 
    createNetworkInterface, 

} from 'apollo-client'; 
const isProduction = process.env.NODE_ENV !== 'development'; 
const testUrl = 'http://localhost:3000/api'; 

// const url = isProduction ? productionUrl : testUrl; 
const url = testUrl; 


const client = new ApolloClient({ 

    networkInterface: createNetworkInterface({uri:testUrl}), 
    dataIdFromObject:({id}) => id, 
    initialState: (typeof window !=='undefined')? window.__STATE__:{}, 
    reduxRootSelector:state => state.custom 

}); 

export default client; 

Home.js

import React,{Component} from 'react'; 
import { connect } from 'react-redux'; 
import { bindActionCreators } from 'redux'; 
import { graphql } from 'react-apollo'; 

import gql from 'graphql-tag'; 

import * as postActions from '../../Redux/Actions/postActions'; 


class Home extends Component{ 
    componentWillMount(){ 
     // console.log('From Will Mount',this.props.posts) 
    } 
    renderAllPost(){ 
     const {loading,posts} = this.props; 

     if(!loading){ 
      return posts.map(data => { 
       return <li key={data.id}>{data.title}</li> 
      }) 
     }else{ 
      return <div>loading</div> 
     } 
    } 
    render(){ 

     return(
      <div> 

       {this.renderAllPost()} 

      </div> 
     ) 
    } 
} 


//start from here 
const GetallPosts = gql` 
query getAllPosts{ 
    posts{ 
    id 
    title 
    body 
    } 
} 
`; 

const mapDispatchToProps = (dispatch) => ({ 
    actions:bindActionCreators(
     postActions, 
     dispatch 
    ) 
}); 


const ContainerWithData = graphql(GetallPosts,{ 
    props:({ data:{loading,posts} }) => ({ 
     posts, 
     loading, 
    }) 
})(Home) 


export default connect(
    // mapStateToPros, 
    // mapDispatchToProps 
)(ContainerWithData) 

回答

1

我可以證實,我理解正確的問題?

您正在渲染HTML服務器端。

  • HTML(包括所有帖子)都在HTML中返回給瀏覽器。
  • 反應,那麼改變這加載窗口
  • 反應,那麼使得API調用,並呈現新的職位

注:阿波羅總會讓AJAX調用,因爲這是由於部分自動完成ContainerWithData。

解決方案 使用所有數據渲染Redux Store。 例如,在調用「createStore」時,您當前正在傳入一個空對象。如果您在此處進行AJAX調用,則可以使用所需的所有數據填充瀏覽器/ redux存儲。

此時,您可以在容器中刪除對GraphQL的調用。你可以用「componentWillMount」替換一些邏輯。

的邏輯是:

  • 創建商店使用從API返回的模型數據
  • 呼叫到家組件
  • 本壘打 「componentWillMount」
  • componentWillMount檢查store.posts有數據
    • 然後從API(GraphQL)加載數據
    • 別的返回true,繼續渲染
+0

你能請提供一些代碼示例.ill給你你想要的數據。 – Nane

相關問題