У меня возникли проблемы с определением наилучшего способа отображения моих реагирующих компонентов только после того, как HTTP-запросы были разрешены.
Например:
компонент A ‹- компонент B ‹- компонент C (C запускает действие, которое вызывает API и возвращает данные для рендеринга, тем временем просто рендерит «загрузку»).
При проверке моего исходного кода я вижу только «загрузку» и хотел бы, чтобы в целях SEO сервер ждал, пока вызов компонента C не будет разрешен и обработан.
Теоретически это звучит просто, потому что сервер должен сам вызвать это действие и дождаться завершения действия, а затем вызвать react.renderToString()
, например:
server.get('/', function (req, res, next) {
showMessages({}, function showMessagesCallback() { //add a callback
var html = React.renderToString(ChatComponent());
res.send(html);
});
});
Но что, если несколько компонентов вызывают действия, и мне нужно дождаться решения нескольких действий, а затем вызвать renderToString
Вместо этого запросы принимаются на стороне клиента. Мой серверный файл:
/**
* This leverages Express to create and run the http server.
* A Fluxible context is created and executes the navigateAction
* based on the URL. Once completed, the store state is dehydrated
* and the application is rendered via React.
*/
import express from 'express';
import path from 'path';
import serialize from 'serialize-javascript';
import {navigateAction} from 'fluxible-router';
import debugLib from 'debug';
import React from 'react';
import app from './app';
import HtmlComponent from 'components/Html';
const htmlComponent = React.createFactory(HtmlComponent);
const debug = debugLib('quran-com');
const server = express();
server.set('state namespace', 'App');
server.use('/public', express.static(path.join(__dirname, '/build')));
server.use('/images', express.static(path.join(__dirname, '/client/images')));
server.use('/fonts', express.static(path.join(__dirname, '/client/styles/fonts')));
server.use((req, res, next) => {
let context = app.createContext();
debug('Executing navigate action');
context.getActionContext().executeAction(navigateAction, {
url: req.url
}, (err) => {
if (err) {
if (err.status && err.status === 404) {
next();
} else {
next(err);
}
return;
}
debug('Exposing context state');
const exposed = 'window.App=' + serialize(app.dehydrate(context)) + ';';
debug('Rendering Application component into html');
const html = React.renderToStaticMarkup(htmlComponent({
context: context.getComponentContext(),
state: exposed,
markup: React.renderToString(context.createElement())
}));
debug('Sending markup');
res.type('html');
res.write('<!DOCTYPE html>' + html);
res.end();
});
});
const port = process.env.PORT || 8000;
server.listen(port);
console.log('Listening on port ' + port);
export default server;
как лучше всего это сделать?