banner



How To Build A Blog Site From Scratch With React

Serverless application architectures are gaining in popularity and it's no mystery why. Developers are able to build and iterate on products faster when they have less infrastructure to maintain and don't need to worry about server maintenance, outages, and scaling bottlenecks.

In this tutorial we are going to show you how to build a serverless, SEO-friendly, CMS-powered blog using React, ButterCMS, and Netlify. The finished code for this tutorial is available on Github.

Getting Started

We'll use the Create React App starter kit.

Install Create React App:

          npm install -g create-react-app                  

Then create the boilerplate for our app:

          create-react-app react-serverless-blog cd react-serverless-blog npm start                  

Creating Routes

Our blog needs two screens: one for listing all posts and another for displaying individual posts. Create BlogHome.js and BlogPost.js components in the src directory:

          import React, { Component } from 'react';  class BlogHome extends Component {   render() {     return (       <div>         Home       </div>     );   } }  export default BlogHome;  import React, { Component } from 'react';  class BlogPost extends Component {   render() {     return (       <div>         Post       </div>     );   } }  export default BlogPost;                  

Create React App doesn't offer routing out-of-the-box so we'll add react-router:

          npm install react-router@3.0.3 --save                  

In the source folder, create a new file called routes.js. We'll create routes for the blog home page with and without page parameters, as well as the individual post page:

          import React from 'react'; import { Router, IndexRoute, Route } from 'react-router';  import App from './App'; import BlogHome from './BlogHome'; import BlogPost from './BlogPost';  const Routes = (props) => (   <Router {...props}>     <Route path="/" component={App}>       <IndexRoute component={BlogHome} />       <Route path="/p/:page" component={BlogHome} />       <Route path="/post/:slug" component={BlogPost} />     </Route>   </Router> );  export default Routes;                  

Next, we'll update index.js so it uses our routes when initializing the application:

          import React from 'react'; import ReactDOM from 'react-dom'; import { browserHistory } from 'react-router';  import Routes from './routes';  ReactDOM.render(  <Routes history={browserHistory} />,  document.getElementById('root') );                  

And finally, we'll update App.js so it nests child components specified in our routes:

          import React, { Component } from 'react';  class App extends Component {   render() {     return (       <div className="App">         <div className="App-header">           <h2>My blog</h2>         </div>         <div>           {this.props.children}         </div>       </div>     );   } }  export default App;                  

Discover how Butter works with any tech stack.

Building the blog

Next we'll slide in ButterCMS to bring our blog to life. ButterCMS provides an API that lets you easily fetch blog posts, categories, tags, and authors.

First we'll install the ButterCMS JS client:

          npm install buttercms --save                  

We'll then update 'BlogHome' to fetch posts from ButterCMS and render them. Use the API token in the example below or get your own by signing into ButterCMS with Github.

          import React, { Component } from 'react'; import { Link } from 'react-router' import Butter from 'buttercms'  const butter = Butter('de55d3f93789d4c5c26fb07445b680e8bca843bd');  class BlogHome extends Component {    constructor(props) {     super(props);      this.state = {       loaded: false     };   }    fetchPosts(page) {     butter.post.list({page: page, page_size: 10}).then((resp) => {       this.setState({         loaded: true,         resp: resp.data       })     });   }    componentWillMount() {     let page = this.props.params.page || 1;      this.fetchPosts(page)   }    componentWillReceiveProps(nextProps) {     this.setState({loaded: false});      let page = nextProps.params.page || 1;      this.fetchPosts(page)   }    render() {     if (this.state.loaded) {       const { next_page, previous_page } = this.state.resp.meta;        return (         <div>           {this.state.resp.data.map((post) => {             return (               <div key={post.slug}>                 <Link to={`/post/${post.slug}`}>{post.title}</Link>               </div>             )           })}            <br />            <div>             {previous_page && <Link to={`/p/${previous_page}`}>Prev</Link>}              {next_page && <Link to={`/p/${next_page}`}>Next</Link>}           </div>         </div>       );     } else {       return (         <div>           Loading...         </div>       )     }   } }  export default BlogHome;                  

Next we'll update BlogPost.js to fetch and display posts based on the route:

          import React, { Component } from 'react'; import Butter from 'buttercms'  const butter = Butter('de55d3f93789d4c5c26fb07445b680e8bca843bd');  class BlogPost extends Component {    constructor(props) {     super(props);      this.state = {       loaded: false     };   }    componentWillMount() {     let slug = this.props.params.slug;      butter.post.retrieve(slug).then((resp) => {       this.setState({         loaded: true,         post: resp.data.data       })     });   }    render() {     if (this.state.loaded) {       const post = this.state.post;        return (         <div>           <h1>{post.title}</h1>           <div dangerouslySetInnerHTML={{__html: post.body}} />         </div>       );     } else {       return (         <div>           Loading...         </div>       );     }   } }  export default BlogPost;                  

Head Elements

Our blog is working but our post pages don't have properly set HTML titles or meta tags. To fix this, we'll use React Helmet.

First, install the package:

          npm install --save react-helmet                  

Import the Helmet module in our BlogPost and use it in our render() method to set HTML tags that should be in the <head>.

                      <Helmet>     <title>{post.seo_title}</title>     <meta name="description" content={post.meta_description} />     <meta name="og:image" content={post.featured_image} />   </Helmet>                  

Here's what the complete code for the component looks like. Inspect the DOM to verify that tags are getting set correctly.

          import React, { Component } from 'react'; import Butter from 'buttercms' import { Helmet } from "react-helmet";  const butter = Butter('de55d3f93789d4c5c26fb07445b680e8bca843bd');  class BlogPost extends Component {    constructor(props) {     super(props);      this.state = {       loaded: false     };   }    componentWillMount() {     let slug = this.props.params.slug;      butter.post.retrieve(slug).then((resp) => {       this.setState({         loaded: true,         post: resp.data.data       })     });   }    render() {     if (this.state.loaded) {       const post = this.state.post;        return (         <div>           <Helmet>             <title>{post.seo_title}</title>             <meta name="description" content={post.meta_description} />             <meta name="og:image" content={post.featured_image} />           </Helmet>            <h1>{post.title}</h1>           <div dangerouslySetInnerHTML={{__html: post.body}} />         </div>       );     } else {       return (         <div>           Loading...         </div>       );     }   } }  export default BlogPost;                  

Discover how Butter works with any tech stack.

Prerendering

Our blog is set up, but crawlers from search engines and social networks don't always execute Javascript so our blog has terrible SEO.

There are a number of ways to address this, including setting up server-side rendering and pre-rendering services like Prerender.io. But these options all require setting up and maintaining a Node.js server. Luckly, Netlify offers built-in prerendering which allows us to keep our code "serverless" while still making sure our blog is optimized for crawlers.

Deploying to Netlify

Before deploying to Netlify we'll create a production build of our app:

          yarn install yarn run build                  

Check your build directory to verify that the production build was created successfully. To upload your site on Netlify, drag the build folder onto Netlify's dashboard.

Prerendering can be enabled from your app settings.

Your blog will work when its loaded from the homepage, but if you try to load a post directly, you'll get a 404 page:

Remember that our app is serverless? To fix the 404 error we need to create a URL rewrite rule that servers up index.html no matter what URL the browser requests.

To do this we'll create a _redirects file in our build directory with the following line:

          /*    /index.html   200                  

Re-upload the contents of your build directory, and all your pages should load. Check out our live example.

To make sure this file gets redirected each time we build our app, we can update our build script in package.json:

          "build": "react-scripts build && echo '/*    /index.html   200' > build/_redirects"                  

Wrap Up

That's it! We've built a SEO-friendly serverless blog using React, Netlify, and ButterCMS. With our serverless approach we can now get back to building software without having to ever worry about server maintenance, outages, or scaling bottlenecks.

I hope you enjoyed this tutorial. If you have any questions about setting up your ButterCMS-powered React app reach out to me at roger@buttercms.com and I'll definitely reply!

Receive tutorials, informative articles, and ButterCMS updates to keep your work running smoothly.

How To Build A Blog Site From Scratch With React

Source: https://buttercms.com/blog/serverless-react-blog-tutorial

Posted by: weaverfromente.blogspot.com

0 Response to "How To Build A Blog Site From Scratch With React"

Post a Comment

Iklan Atas Artikel

Iklan Tengah Artikel 1

Iklan Tengah Artikel 2

Iklan Bawah Artikel