For those node-heads out there, React Static provides a Node API for full programmatic control!
The following exports are available via the react-static/node
module:
We recommended using the latest ES6 import syntax:
import { create, start, build } from 'react-static/node'
Creates a new react-static project.
- Arguments
name: string
- The name of your new project (relative to the current working directory), or the full path to the new directory you wish to createlocation: string
- The name of the template in the
examples
directory - The full URL of a public git repository
- The full path to a local directory
- The name of the template in the
- Returns a
Promise
Starts the development server.
- Arguments
config: object || string
- The config object to use, or the path of thestatic.config.js
file you wish to use.
- Returns a
Promise
that will never resolve. The process must be exited by the user to stop the server.
Builds your site for production. Outputs to a dist
directory in your project.
- Arguments
config: object || string
- The config object to use, or the path of thestatic.config.js
file you wish to use.staging
- Whentrue
, no siteRoot replacement or absolute URL optimizations are performed, allowing a production build of your site to function on localhost more easily. Use this argument to test a production build locally.debug
- Whentrue
, your build will not beuglified
allowing you to debug production errors (as long as they are unrelated to minification or uglification)
- Returns a
Promise
Intended for use in your static.config.js
during development. When called it will rebuild all of your your routes and routeData by calling config.getRoutes()
again. Any new routes or data returned will be hot-reloaded into your running development application. Its main use cases are very applicable if your routes or routeData are changing constantly during development and you do not want to restart the dev server. You can use this method to reload when local files are changed, update at a set timing interval, or even subscribe to an event stream from an API or CMS.
Example:
// static.config.js
import { reloadRoutes } from 'react-static/node'
// Reload Manually
reloadRoutes()
// Reload when files change
import chokidar from 'chokidar'
chokidar.watch('./docs').on('all', () => reloadRoutes())
// Reload from API or CMS event
YourFavoriteCMS.subscribe(reloadRoutes)
// Reload your routes every 10 seconds
setInterval(reloadRoutes, 10 * 1000)
// ETC!
export default {
getRoutes: () => {
// This will run each time `reloadRoutes` is called
},
}
A utility function to aid in splitting an array of items into separate pages for use in your static.config.js
- Arguments
options{}
- Requireditems: Array
- Required - The array of items to split into pagespageSize: Int
- Required - The number of items on each pageroute{}: Object
- Requiredpath: String
- Required - The base path that all pages will sharecomponent: String
- The base component that all pages will share
decorate: Function
- Required- Arguments:
items: Array
- The items for the given pagepageIndex: Int
- The page index for the given pagetotalPages: Int
- The total number of pages that were generated
- Returns an
Object
that will decorate the base route. In most cases, this will probably include thegetData
andchildren
keys, but can contain any route supported keys
- Arguments:
pageToken: String
- The string that will be used to prefix each page.
- Returns an array of routes objects
Example:
// static.config.js
import { makePageRoutes } from 'react-static/node'
export default {
getRoutes: () => {
const posts = [...]
return [
...makePageRoutes({
items: posts, // Use the posts array as items
pageSize: 5, // Use 5 items per page
pageToken: 'page', // use page for the prefix, eg. blog/page/3
route: {
// Use this route as the base route
path: 'blog',
component: 'src/containers/Blog',
},
decorate: (items, pageIndex, totalPages) => ({
// For each page, supply the posts, page and totalPages
getData: () => ({
posts: items,
currentPage: pageIndex,
totalPages,
}),
// Make the routes for each blog post
children: items.map(post => ({
path: `/blog/post/${post.id}`,
component: 'src/containers/Post',
getData: () => ({
post,
}),
})),
}),
}),
]
}
}
Each route's getData
function results in a separate data file for each route being stored as JSON next to the routes HTML on export. This covers the 90% use case for data splitting, but if you want even more control and want to optimize repeated data across routes, you can use this function to create shared data fragments for use in your routes.
These shared data fragments can be placed in any route's sharedData
object. At runtime, the shared data will only be requested once per session and automatically merged into the route data, which you can consume normally through the RouteData
or withRouteData
component/HOC pair.
Example
Consider a large and heavy menu structure that is present only on the blog portion of your site. In this case, the menu data should only be loaded on the pages that use it, and only once per session (cached), instead of on every page individually. First we would use the createSharedData
function and pass the data we want to share between our routes. Then in each route, we can pass the result of our createSharedData
call as a prop to the route's sharedData
property. The shared data props will then be stored, served and cached only once per session and merged into the result of the routes getData
result at runtime!
import axios from 'axios'
export default {
getRoutes: async () => {
const blogMenu = getMyDynamicMenu()
return [
{
path: '/',
component: 'src/containers/Home',
},
{
path: '/blog',
component: 'src/containers/Docs',
sharedData: {
blogMenu, // `blogMenu` will now be available to this route via
// RouteData but will only be loaded once per session!
},
},
{
path: '/help',
component: 'src/containers/Help',
sharedData: {
blogMenu, // `blogMenu` will now be available to this route via
// RouteData but will only be loaded once per session!
},
},
]
},
}