Come with me on a short adventure as we build a fast, simple interface for publishing blog style articles. Something sensible. It should look clean and simple yet have the ability to perform powerful resource fetching with some query magic. I’m talking about Graphql, the query language built with web APIs in mind. And speaking of web APIs, we will also be using our handy Content Management Service: Cosmic JS, to Create and Store our Blog Data.
Progressive Apollo Blog
Progressive Apollo Blog Codebase
Before you go any further make sure you have the required developer tools installed on your machine. Mainly you will need Node JS, its accompanying package manager: npm, and git. Git is semi optional and will only be used to code storage and deployment if that is your goal.
Once you have those installed we can begin setting up our project and eventually start writing some Javascript.
Let’s go over the main packages that we will use to create our blog platform.
I believe with all that boring business out of the way, we can start slapping our keyboards!
Let’s fire up our terminal and start bootstrapping our project source files. For those that don’t have Create React App installed, we can do that with a simple npm command:
$npm install -g create-react-app
Once this finishes (or you have this package already installed), we can summon our initial project files by running the cli:
$create-react-app apollo-blog
You will notice a fair amount of wizardry occurring in your terminal as our app is bootstrapped with all sorts of powerful libraries and source code. Taking a look at our file system will show a directory titled apollo-blog
. Changing directories into this folder will show a structure similar to this:
apollo-blog/ README.md node_modules/ package.json public/ index.html favicon.ico src/ App.css App.js App.test.js index.css index.js logo.svg
We should now be able to run:
$npm start
or
$yarn start
This will run the start script stored in our `package.json` file which is all bundled together with Webpack and build tools that will handle compiling all of our React source code and bundling that into files ready to be served to a browser.
After the script runs you should see a simple message in your terminal, and your default browser should open to a tab with project running. You should see something like this:
The next step is installing some npm libraries so that will help quickly create a good looking App Interface as well as easily fetch data from the Cosmic JS graphql servers.
In your terminal run the following command and let’s talk about what each of these packages will be doing for us:
$npm install --save apollo-boost react-apollo graphql dotenv semantic-ui-css semantic-ui-react react-router-dom
apollo-boost / react-apollo / graphql — These guys are going to serve as the main solution for getting our data efficiently and scalably from Cosmic JS servers to our client side UI. They provide wrappers for our components to be able to configure queries and provide useful variables for data loading and error handling.
dotenv — this package will allow is to configure semi sensitive keys so we don’t have to store anything publicly for accessing our Cosmic JS Bucket.
semantic-ui-css / semantic-ui-react — Our UI framework that will let us import pre-styled containers and components letting us focus on programming functionality, spending less time fidgeting with CSS and other styling considerations.
react-router-dom — A react library that will allow us to configure url routing logic to control linking within our app and accessing URL parameters.
with these installed we can begin writing some javascript!
We are going to add a minimal security precaution so that we won’t let just any request come through our bucket’s API endpoint, feel free to skip this section but be warned that your components will behave a little differently and your bucket can be queried from anyone with the knowhow to make API Requests. So, you know… don’t put any sensitive info in your buckets.
We are going to require a read_key to be attached to every API request to our bucket. You’ll have to generate one by heading to the settings tab for your Cosmic JS bucket and clicking on Basic Settings. Cosmic JS > Your Bucket > settings > main.
At the top there should be as section titled API Access > API Read Access Key. Click on the button to Generate new key. Copy the key to your clipboard and let’s configure our project directory with our config file:
$touch .env
this file is going to be used to configure our project with our API Read Access Key so we don’t have to hard code it in with our source code and commit it for the world / other prying eyes to see.
simply add one line to your .env file:
REACT_APP_COSMIC_JS_READ_KEY=your_cosmic_js_read_key
Make sure you prefix your variable name with REACT_APP! The build scripts that bundle our source code will look for this to make variables available in our compiled javascript. With that done, we are now ready to use this key in our React project.
Finally, we are ready to create our Components and set up our routes. In an effort to keep things as simple as possible we are just creating two routes and two main components:
Routes
{protocol}://{host}/
{protocol}://{host}/article/:articleName
Components
/views/home.js
/views/article.js
Let’s go ahead and open the App.js component in whatever text editor you prefer, and start to configure it with our routing logic and our Graphql client.
We are going to import one component wrapper from react-apollo and one constructor function that will let the components within our component wrapper make requests to a specified Graphql server ( for us this will be the Cosmic JS’s Graphql server).
We also want to import our wrapping components for react-router-dom. This will allow us to us our browser url to systematically switch component views and access any url parameters.
Lastly we are going to import a CSS file from Semantic UI so that we can leverage it’s class selectors and give imported components their base stylings.
Here’s what our App.js should look like after everything is properly configured:
import React, { Component } from 'react' import ApolloClient from 'apollo-boost' import { ApolloProvider } from 'react-apollo' import { BrowserRouter, Route } from 'react-router-dom' import './App.css' import 'semantic-ui-css/semantic.min.css'import Home from './views/home' import Article from './views/article' // creates our Graphql client using the CosmicJS Graphql endpoint const client = new ApolloClient({ uri: "https://graphql.cosmicjs.com/v1" }) class App extends Component { render() { return ( <ApolloProvider client={client}> <BrowserRouter> <div className="router-container"> <Route exact path="/" component={Home} /> <Route path="/article/:articleName" component={Article} /> </div> </BrowserRouter> </ApolloProvider> ) } } export default App
Let’s create a new file in a new directory
src/views/home.js
This folder will store all of our view layer React Components. Our Home View will just return some brief info about our site and a list of articles that have been published to our blog.
The main parts of this component are a heading that display a title and a site description, and a list that will display Semantic UI components.
We also will need to configure this view with some Graphql logic so that it can make requests to Cosmic JS
import React from ‘react’
import gql from ‘graphql-tag’
import { graphql } from ‘react-apollo’
import { Link } from ‘react-router-dom’// Component from Semantic UI import { Container, Image, Card, Label, Loader } from 'semantic-ui-react' const GET_ARTICLES = gql` query Articles($read_key: String!) { objectsByType(bucket_slug: "apollo-blog", type_slug: "articles", read_key: $read_key ) { _id slug created_at title type_slug metadata } } ` class Home extends React.Component { render() { const { data: { loading, error, objectsByType } } = this.props const styles = { heading: { margin: '0 0 20px 0', height: '300px', display: 'flex', flexDirection: 'column', justifyContent: 'center', alignItems: 'center', fontFamily: "'Rubik', sans-serif", backgroundColor: '#4ABDAC', color: 'white', }, headingTitle: { marginBottom: '30px', display: 'flex', flexDirection: 'row', justifyContent: 'center', alignItems: 'center', }, container: { minHeight: 'calc(100vh - 500px)', display: 'flex', flexDirection: 'row', justifyContent: 'center', alignItems: 'flex-start', flexWrap: 'wrap', }, title: { fontSize: '200%', margin: '10px 0', }, card: { minWidth: '350px', }, summary: { fontSize: '120%', lineHeight: '25px', } } return ( <Container className="home-container" style={styles.container}> <div className="home-heading" style={styles.heading}> <div style={styles.headingTitle}> <h1>Apollo Blog</h1> <p>A simple and extensible blog platform</p> </div> </div> {loading ? <Loader active inline="centered" size="massive">...loading</Loader> : null} {error ? `Error! ${error.message}` : null} {objectsByType ? <Container className="articleList-container" style={styles.container}> {objectsByType.map(article => { let { summary, image, categories, author } = article.metadata if (categories) categories = categories.split(' ') return ( <Link key={article._id} to={`/article/${article.slug}`}> <Card style={styles.card}> <Card.Content> {image ? <Image src={image.url} /> : null} <Card.Header style={styles.title}>{article.title}</Card.Header> <Card.Meta> {author ? author.name : null} <span className='date'>{article.created_at}</span> </Card.Meta> <Card.Description style={styles.summary}>{summary}</Card.Description> </Card.Content> {categories ? <Card.Content extra> {categories.map(category => <Label key={category}>{category.replace(/,/g, '')}</Label>)} </Card.Content> : null } </Card> </Link> ) })} </Container> : null } </Container> ) } } export default graphql(GET_ARTICLES, { options: { variables: { read_key: process.env.REACT_APP_COSMIC_JS_READ_KEY, } }, props: ({ data }) => ({ data, }), })(Home)
Now we have ye old React Component, that will fetch articles from our content manager (Cosmic JS) before the component renders and pass those as an array on component props.
Apollo also gives us some handy properties from is data object, specifically an error object, and a loading boolean, which we use to conditionally render components, such as loading message when the request is fetching, and an error message when our error object returns a message.
With our list of articles displaying on our home page, we need to Create the view that will display the blog content contained in each article object.
Let’s create another file in our views directory:
src/views/article.js
Creating this component is similar to the Home View, but our query is going to include an article identifier from our URL stored on a prop: match which react-router-dom creates for us when we render a component at a Route.
import React from ‘react’
import { Link } from ‘react-router-dom’
import gql from ‘graphql-tag’
import { Query } from ‘react-apollo’
import { Container, Loader, Label, Image } from ‘semantic-ui-react’
import ‘./_article.css’const GET_ARTICLE_BY_SLUG = gql` query Article($read_key: String!, $slug: String!) { object(bucket_slug: "apollo-blog", read_key: $read_key, slug: $slug ) { _id created_at title content metadata } } ` const Article = ({ match }) => { const read_key = process.env.REACT_APP_COSMIC_JS_READ_KEY const article_slug = match.params.articleName const styles = { container: { display: 'flex', flexDirection: 'row', justifyContent: 'center', alignItems: 'flex-start', }, articleContainer: { width: '95%', maxWidth: '1000px', margin: '20px auto', display: 'flex', flexDirection: 'column', justifyContent: 'flex-start', alignItems: 'center', }, article: { width: '70%', margin: '-20px 0 0 20px', fontFamily: '"Quicksand", sans-serif', }, heading: { height: '100px', width: '100%', display: 'flex', flexDirection: 'row', justifyContent: 'center', alignItems: 'center', backgroundColor: '#4ABDAC', }, headingTitle: { color: 'white', }, details: { width: '100%', minHeight: '200px', display: 'flex', flexDirection: 'column', justifyContent: 'center', alignItems: 'center' }, lineOne: { width: '95%', maxWidth: '900px', display: 'flex', flexDirection: 'row', justifyContent: 'flex-start', }, title: { width: '50%', fontSize: '250%', textAlign: 'right', margin: 'auto 20px', lineHeight: '35px', }, date: { fontSize: '130%', lineHeight: '20px', }, image: { maxWidth: '950px', maxHeight: '500px', }, } return ( <Query query={GET_ARTICLE_BY_SLUG} variables={{ read_key, slug: article_slug }}> {({ loading, error, data }) => { if (loading) return <Loader active inline="centered" size="massive">...loading</Loader> if (error) return `Error! ${error.message}` let { image, author, categories } = data.object.metadata if (categories) categories = categories.split(' ') return ( <Container> <div className="post-heading" style={styles.heading}> <Link to="/" className="header-btn"> <h2>Apollo Blog</h2> </Link> </div> <div className="post-details" style={styles.details}> <div style={styles.lineOne}> <h4 className="post-title" style={styles.title}>{data.object.title}</h4> <h6 className="post-date" style={styles.date}>{data.object.created_at}</h6> </div> {categories ? <div> {categories.map(category => <Label key={category}>{category.replace(/,/g, '')}</Label>)} </div> : null } {image ? <Image src={image.url} style={styles.image} /> : null} </div> <div style={styles.articleContainer}> {author ? <h3>By {author.title}</h3> : null} <div className="article" dangerouslySetInnerHTML={{ __html: data.object.content }} style={styles.article} /> </div> </Container> ) }} </Query> ) } export default Article
This component is much simpler than our Home component, mostly due to the fact that we are just using the Query component wrapper instead of turning our component into a higher level component using the Graphqlcomponent. This gives us access to our Graphql directly in our component logic but does not make it available on component props! So take note.
For both components I have included some minimal styling objects in the source code, but feel free to use your creativity and change up the styling properties to create your own look and feel.
And that’s it! We have built two relatively simple and scalable React components that are configured to two separate url routes. We can control our views easily with react-router-dom and fetch our data sensibly with the power of Apollo / Graphql and Cosmic JS.
create-react-app comes bundled with some handy deployment scripts if you feel so inclined to make your application available to the world.
If you take a look at the package.json file that we have a script called build. Running this will compile all your project files and bundle them into a static build directory that is ready to deploy to any hosting service.
You’ll also notice after the script finishes that you have an option to publish directly to github pages using the serve module. You just need to add a homepage value in package.json and make sure the serve package is installed with either npm or yarn.
Follow the link at the end of the build message to learn more about deployment using this method.
Netlify provides app building and deployment services and they will give you a free ( if not a little bit verbose ) URL for you to deploy client side code. The process is pretty easy and we can use a github repository to deploy our code continuously when we make changes to our source code.
After you create and account / Log in to the Netlify dashboard, there should be a tab called sites.
Clicking the sites tab will reveal a list of sites you have deployed on Netlify. From here you have two options:
Deploying the static build directory is super simple but will require you to manually build and upload files to Netlify. Simply drag the build directory directly over the area at the bottom of the sites tab from the Netlify Dashboard.
This will automatically deploy your site to a url that Netlify will generate. Simple but requires work every time you update your source code.
On the far right side there will be a button that reads New site from Git.Selecting this option will walk you through the process of authorizing Netlify to access a code repository. After these steps are finished Netlify will build your site files from a build script you specify and deploy it automatically each time you make a commit to your repo!
Well that’s it for me today. Happy hacking until we meet again :)
Originally published on https://hackernoon.com
#javascript #react-native #graphql