back to all posts

Create a twitch clone using React — 4

by Nabendu Biswas / May 4th, 2019

#javascript #react #webdev
Series: Twitch-react

Welcome to Part-4 of the series. We will start implementing streams in our project.

Our StructureOur Structure

We will now setup the API Server. Now, open the terminal and goto your streams folder. Here create a new folder api and cd into it.

api folderapi folder

Now type npm init in the command line. It will ask you for details of many things. Press enter for each of them.

npm initnpm init

At the end it will show the package.json structure and ask for our permission. Type yes in it.

YesYes

Now, we have a package.json file inside our api folder. Now, we will install the npm package json-server.

json-serverjson-server

Now, open the api directory in VSCode. Here add a db.json file and also a .gitignore file(If you are using github).

api folderapi folder

Add the following in db.json file.

    {
        "streams": []
    }

Now, add the following in .gitignore file

    # See https://help.github.com/articles/ignoring-files/ for more about ignoring files.

    # dependencies
    /node_modules
    /.pnp
    .pnp.js

    # testing
    /coverage

    # production
    /build

    # misc
    .DS_Store
    .env.local
    .env.development.local
    .env.test.local
    .env.production.local

    npm-debug.log*
    yarn-debug.log*
    yarn-error.log*

Now, open the package.json file and update the start script.

    {
        "name": "api",
        "version": "1.0.0",
        "description": "",
        "main": "index.js",
        "scripts": {
            "start": "json-server -p 3001 -w db.json"
        },
        "author": "",
        "license": "ISC",
        "dependencies": {
            "json-server": "^0.14.2"
        }
    }

Now, goto the api folder in terminal and do a npm start.

npm startnpm start

Now, let’s move to our client code again. So, open the terminal where client is running and stop it(Ctrl+c) and then install axios and redux-thunk. Don’t forget to start back the client server after installation with npm start.

axios and thunkaxios and thunk

Now, create a new folder apis inside src. Create a new file streams.js in it.

streams.jsstreams.js

Add the following code in streams.js

    import axios from 'axios';

    export default axios.create({
        baseURL: 'http://localhost:3001'
    });

Now, we will create our action creator in index.js inside action folder. Here we are first import our axios streams. Then creating an action creator to do a post call to /streams endpoint, with formValues.

    import { SIGN_IN, SIGN_OUT } from './types';
    import streams from '../apis/streams';

    export const signIn = (userId) => {
        return {
            type: SIGN_IN,
            payload: userId
        };
    };

    export const signOut = () => {
        return {
            type: SIGN_OUT
        };
    };

    export const createStream = formValues => async dispatch => {
        streams.post('/streams', formValues);
    }

Now, head over to our StreamCreate.js to use this action creator. We are first importing connect and our action createStream . Then we are changing our reduxForm at the end of the file a bit, to include connect.

Finally we are using createStream inside onSubmitForm(), to send formValues to action creator.

    import React, {Fragment} from 'react';
    import { Field, reduxForm } from 'redux-form';
    import { connect } from 'react-redux';
    import { createStream } from '../../actions';
    import './formStyles.css';

    class StreamCreate extends React.Component {
    …
    …

        onSubmitForm = (formValues) => {
            this.props.createStream(formValues);
        }

        render() {
            return (
            <form onSubmit={this.props.handleSubmit(this.onSubmitForm)} className="ui__form">
                <Field name="title" label="Enter Title" component={this.renderInputBox} />
                <Field name="description" label="Enter Description" component={this.renderInputBox} />
                <button>Submit</button>
            </form>
            )
        }
    }

    …
    …

    const formWrapped = reduxForm({form: 'streamCreate', validate})(StreamCreate);

    export default connect(null, { createStream })(formWrapped);

Next, we also have to include redux-thunk in our project. We do the same by going to our root index.js and adding the things marked in bold.

    import React from 'react';
    import ReactDOM from 'react-dom';
    import App from './components/App';
    import { Provider } from 'react-redux';
    import { createStore, applyMiddleware, compose } from 'redux';
    import reducers from './reducers';
    import reduxThunk from 'redux-thunk';

    const composeEnhancers = window.__REDUX_DEVTOOLS_EXTENSION_COMPOSE__ || compose;
    const store = createStore(reducers, composeEnhancers(applyMiddleware(reduxThunk)));

    ReactDOM.render(
    <Provider store={store}><App /></Provider>, document.querySelector('#root'));

Now, time to check our POST request. Head to http://localhost:3000/streams/create and enter data and click on Submit button. Also, keep the network tab in developer console open. You can see the POST request sucessfully made.

The post requestThe post request

We will also open the db.json to verify that the POST request was done.

db.jsondb.json

Now, we will update action creator in index.js inside action folder. We will also add action creators for other RESTful items like DELETE, EDIT, FETCH.

First add those types inside types.js inside action folder.

    export const SIGN_IN = 'SIGN_IN';
    export const SIGN_OUT = 'SIGN_OUT';
    export const CREATE_STREAM = 'CREATE_STREAM';
    export const FETCH_STREAM = 'FETCH_STREAM';
    export const FETCH_STREAMS = 'FETCH_STREAMS';
    export const DELETE_STREAM = 'DELETE_STREAM';
    export const EDIT_STREAM = 'EDIT_STREAM';

Next, add function for the above types in index.js inside action folder.

    import { SIGN_IN, SIGN_OUT, CREATE_STREAM, FETCH_STREAM, FETCH_STREAMS, DELETE_STREAM, EDIT_STREAM } from './types';
    import streams from '../apis/streams';

    export const signIn = (userId) => {
        return {
            type: SIGN_IN,
            payload: userId
        };
    };

    export const signOut = () => {
        return {
            type: SIGN_OUT
        };
    };

    export const createStream = formValues => async dispatch => {
        const response = await streams.post('/streams', formValues);
        dispatch({ type: CREATE_STREAM, payload: response.data })
    }

    export const fetchStreams = () => async dispatch => {
        const response = await streams.get('/streams');
        dispatch({ type: FETCH_STREAMS, payload: response.data })
    }

    export const fetchStream = (id) => async dispatch => {
        const response = await streams.get(`/streams/${id}`);
        dispatch({ type: FETCH_STREAM, payload: response.data })
    }

    export const editStream = (id, formValues) => async dispatch => {
        const response = await streams.put(`/streams/${id}`, formValues);
        dispatch({ type: EDIT_STREAM, payload: response.data })
    }

    export const deleteStream = (id) => async dispatch => {
        await streams.delete(`/streams/${id}`);
        dispatch({ type: DELETE_STREAM, payload: id })
    }

We will now work on our reducer and put it in one go. Create a new file streamReducer.js inside reducer folder.

streamReducer.js filestreamReducer.js file

First let create case for FETCH_STREAM, CREATE_STREAM and EDIT_STREAM in our reducer file streamReducer.js as they are similar.

    import { CREATE_STREAM, FETCH_STREAM, FETCH_STREAMS, DELETE_STREAM, EDIT_STREAM } from '../actions/types';

    export default (state={}, action) => {
        switch(action.type) {
            case FETCH_STREAM:
                return { ...state, [action.payload.id]: action.payload }
            case CREATE_STREAM:
                return { ...state, [action.payload.id]: action.payload }
            case EDIT_STREAM:
                return { ...state, [action.payload.id]: action.payload }
            default:
                return state;
        }
    }

Now, we will create FETCH_STREAMS and DELETE_STREAM case. For that we will first install lodash, which have a very useful omit function.

npm installnpm install

Again open the streamReducer.js inside reducer folder.

    import { CREATE_STREAM, FETCH_STREAM, FETCH_STREAMS, DELETE_STREAM, EDIT_STREAM } from '../actions/types';
    import _ from 'lodash';

    export default (state={}, action) => {
        switch(action.type) {
            case FETCH_STREAMS:
                return { ...state, ..._.mapKeys(action.payload, 'id')};**
            case FETCH_STREAM:
                return { ...state, [action.payload.id]: action.payload };
            case CREATE_STREAM:
                return { ...state, [action.payload.id]: action.payload };
            case EDIT_STREAM:
                return { ...state, [action.payload.id]: action.payload };
            case DELETE_STREAM:
                return _.omit(state, action.payload);**
            default:
                return state;
        }
    }

Next, we will hook this new reducer to our root reducer. So, open index.js inside reducer folder.

    import { combineReducers } from 'redux';
    import authReducer from './authReducer';
    import streamReducer from './streamReducer';
    import { reducer as formReducer } from 'redux-form';

    export default combineReducers({
        auth: authReducer,
        form: formReducer,
        streams: streamReducer
    });

We will now start to show the streams created by us. Open StreamList.js which is inside components -> streams folder and hook our action creator.

    import React from 'react';
    import { connect } from 'react-redux';
    import { fetchStreams } from '../../actions';

    class StreamList extends React.Component {
        componentDidMount() {
            this.props.fetchStreams();
        }

        render(){
                return (
                    <div>
                    StreamList
                    </div>
            )
        }
    }

    export default connect(null, { fetchStreams })(StreamList);

Now, go to http://localhost:3000/ and open the developer console. Goto the Redux tab. You can see the stream, we created earlier here.

Stream created earlierStream created earlier

Before moving forward, let’s create another stream So, open http://localhost:3000/streams/create and create another stream.

Another streamAnother stream

Next, we will show these streams in our home page. So, go ahead and open StreamList.js which is inside components -> streams folder and add below. Here, we are getting the data in streams object. We are then iterating over it and showing it through a renderList method.

    import React from 'react';
    import { connect } from 'react-redux';
    import { fetchStreams } from '../../actions';

    class StreamList extends React.Component {
        componentDidMount() {
            this.props.fetchStreams();
        }

        renderList() {
            return this.props.streams.map(stream => {
            return (
                <div className="item" key={stream.id}>
                <div className="content">{stream.title}</div>
                <div className="description">{stream.description}</div>
                <hr />
                </div>
            )
            })
        }

        render() {
            return (
            <div>
                <h2 className="header">Streams</h2>
                <div className="list">{this.renderList()}</div>
            </div>
            )
        }
    }

    const mapStateToProps = (state) => {
        return { streams: Object.values(state.streams) };
    }

    export default connect(mapStateToProps, { fetchStreams })(StreamList);

Right now if you look into our Home page, the list is been show but there is no style.

Not much styleNot much style

Next, we will add some styles to it to make it look good. Create a new file StreamList.css inside same components -> streams folder. Don’t forget to import it first in our StreamList.js file.

    import React from 'react';
    import { connect } from 'react-redux';
    import { fetchStreams } from '../../actions';
    import './StreamList.css';
    ...
    ...

Now in StreamList.css add the following code.

    .list{
        display: grid;
        grid-gap: 1.2rem;
    }

    .content{
        font-size:1.2rem;
        font-weight: bold;
        margin-bottom: 5px;
    }

    .description{
        font-size:1.1rem;
    }

    .header{
        background: #E6F2F5;
        padding: 10px;
    }

Now, our app looks like below.

Good looking appGood looking app

This concludes part-4 of the series. You can find code till this point here.

Nabendu Biswas