Hey, thanks for taking the time to read my repo. I tried to make it as clear as possible. Slack me (@jameslemire) if you think something could be explained better. Also, if you learn better by looking at code, check out the other branch called FinishedProject.
Also please note that this repo is just to explain the absolute minimum of making CRUD calls, it's not intended as the "best practice" way of using React with CRUD.
If you understand how React with CRUD works check out the IndustryStandard page that goes into the a 'industry standard' way of using these concepts. I didn't include instructions for that one. It's pretty complicated...
- I highly recommend you copy the package.json and webpack.config.js files and put them in a "resource" folder on your desktop. They are setup to be put into any new project We are going to modify the package.json and webpack.config.js files so they are connected to this project's current file structure
- In your package.json we have the option to edit the name, description, and author keys (It isn't necessary) They are located on lines 2, 4, and 9 -- However the webpack.config.js file changes ARE necessary. This file tells webpack where to output bundle.js and the entry point for webpack.
- In your webpack.config.js change line 3 to match the location of your reactApp.js file. The file location is based off of webpack.config.js' location. ( './public/app/reactApp.js' )
- In your webpack.config.js change line 7 to match the location of your scripts folder. The file location is based off of webpack.config.js location. ( './public/script' )
1 - Every react app, at the bare minimum needs, React and ReactDOM imported. On the very first two lines of code import them. Also for this projet we'll need axios to make backend calls, import that as well.
import React from 'react';
import ReactDOM from 'react-dom';
import axios from 'axios';
2 - We're now ready to create our App using var.
var App = React.createClass({
})
3 - React apps need a render method that returns the html that gets put on the page. Create a render method for our app
var App = React.createClass({
render() {
return (
<div>
<p>Hello World</p>
</div>
)
}
})
4 - To initialize our React application you have to use ReactDOM with the render method.
ReactDOM.render(<App />, document.getElementById('app'));
Now the basics of our react app are ready to go. We have to use 'app' because our initialize render method is looking for a document with element id of 'app'.
5 - Setup a basic html file and include at least one div in the body with the id of 'app'. In addition include a script tag with a src attribute equal to the directory of where webpack will send bundle.js
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<title>React CRUD</title>
</head>
<body>
<div id='app'></div>
<script src="./script/bundle.js"></script>
</body>
</html>
6 - Remember to run webpack to see any react changes or run webpack -w
so webpack will re-run when you edit the reactApp.js file
7 - Live-server or open your index.html and you should see Hello World on the page
1 - Using node create a basic backend. Make sure to install express, body-parser, massive and cors at the bare minimum
npm install --save express body-parser massive cors
2 - Your server.js file should look similar to this after you're done setting up
var express = require('express');
var bodyParser = require('body-parser');
var cors = require('cors');
var massive = require('massive');
var connectionString = 'postgress://jameslemire@localhost/sandbox'
var massiveInstance = massive.connectSync({connectionString : connectionString});
var app = express();
app.set('db', massiveInstance);
app.use(cors({origin: 'http://localhost:3000/'}));
app.use(bodyParser.json());
app.use(express.static('../public'));
app.listen(3000, function() { console.log('Server started on port 3000'); });
3 - Create a table named people CREATE TABLE people (id SERIAL PRIMARY KEY, first_name VARCHAR(50), last_name VARCHAR(50) );
4 - Don't forget to make a db folder in the same directory as your server.js file
#Setting up your endpoints Now we're ready to make some CRUD endpoints in our server. (Create, Read, Update, Delete)
1 - Create four basic end points that follow CRUD with four sql commands. Your code should look similar to this
app.post('/api/add/person', function(req, res) {
db.addPerson([req.body.first_name, req.body.last_name], function(err, r) {
res.status(200).send('Person created');
});
})
app.get('/api/people', function(req, res) {
db.getPeople(function(err, r) {
res.json(r);
});
});
app.post('/api/update/person', function(req, res) {
db.updatePerson([req.body.id, req.body.first_name, req.body.last_name], function(err, r) {
res.status(200).send('Person updated');
})
})
app.delete('/api/person/:personId', function(req, res) {
db.deletePerson([req.params.personId], function(err, r) {
res.status(200).send('Person deleted');
})
})
addPerson.sql
INSERT INTO people (first_name, last_name) VALUES ($1, $2);
getPeople.sql
SELECT * FROM people;
updatePerson.sql
UPDATE people
SET first_name = $2, last_name = $3
WHERE id = $1;
deletePerson.sql
DELETE FROM people WHERE id = $1;
#Server file structure Your server file structure should look similar to this
--private
--db
-addPerson.sql
-deletePerson.sql
-getPeople.sql
-updatePerson.sql
--node_modules
-package.json
-server.js
#Friendly Reminder Don't forget to pack your webz (shoutout to Heather). Meaning don't forget to run webpack when you're trying to see changes you make on your react app. Also remember we need our server running to make the api calls. Which means we also need postgres running. Which means live-server will no longer be of use because you need to be on the address that you're using for express.static. (What a handful..) #Setting up the front-end In order to get our endpoints to work with our front end we are going to need some inputs and a button
1 - Add two inputs (first name, last name) and a button (add person)
import React from 'react';
import ReactDOM from 'react-dom';
import axios from 'axios';
var App = React.createClass({
render() {
return (
<div>
<h1>Add Person</h1>
<input type="text" />
<input type="text" />
<button>Add Person</button>
</div>
)
}
})
ReactDOM.render(<App />, document.getElementById('app'));
2 - Using getInitialState() add variables to store the array of people, first_name input, and last_name input.
getInitialState() {
return {
first_name: '',
last_name: '',
people: []
}
},
3 - Using onChange={}
update these variables while the user types in either input.
<input onChange={this.first_nameCatcher} type="text" />
<input onChange={this.last_nameCatcher} type="text" />
first_nameCatcher(event) {
this.setState({
first_name: event.target.value
})
},
last_nameCatcher(event) {
this.setState({
last_name: event.target.value
})
},
Your code will now look something like this
import React from 'react';
import ReactDOM from 'react-dom';
import axios from 'axios';
var App = React.createClass({
getInitialState() {
return {
first_name: '',
last_name: '',
people: []
}
},
render() {
return (
<div>
<h1>Create Person</h1>
<input onChange={this.first_nameCatcher} type="text" />
<input onChange={this.last_nameCatcher} type="text" />
<button>Add Person</button>
</div>
)
},
first_nameCatcher(event) {
this.setState({
first_name: event.target.value
})
},
last_nameCatcher(event) {
this.setState({
last_name: event.target.value
})
}
})
ReactDOM.render(<App />, document.getElementById('app'));
#CRUD - Create
1 - Using onClick={}
make an api call to add a person. Make sure to pass the data in the call, axios works just like $http (how freaking cool!)
<button onClick={this.createPerson}>Create Person</button>
createPerson() {
var person = {first_name: this.state.first_name, last_name: this.state.last_name}
axios({
method: 'POST',
url: 'http://localhost:3000/api/add/person',
data: person
})
}
Your code should now look something like this
import React from 'react';
import ReactDOM from 'react-dom';
import axios from 'axios';
var App = React.createClass({
getInitialState() {
return {
first_name: '',
last_name: '',
people: []
}
},
render() {
return (
<div>
<h1>Create Person</h1>
<input onChange={this.first_nameCatcher} type="text" />
<input onChange={this.last_nameCatcher} type="text" />
<button onClick={this.createPerson}>Create Person</button>
</div>
)
},
first_nameCatcher(event) {
this.setState({
first_name: event.target.value
})
},
last_nameCatcher(event) {
this.setState({
last_name: event.target.value
})
},
createPerson() {
var person = {first_name: this.state.first_name, last_name: this.state.last_name}
axios({
method: 'POST',
url: 'http://localhost:3000/api/add/person',
data: person
})
}
})
ReactDOM.render(<App />, document.getElementById('app'));
#CRUD - Read
1 - To read our database we can use another onClick={}
to make an api call to our backend. Then using a map function we can display the users it finds
<button onClick={this.readPeople}>Read People</button>
readPeople() {
axios({
method: 'GET',
url: 'http://localhost:3000/api/people'
}).then(r => {
this.setState({
people: r.data
})
});
}
{this.state.people.map( (value, index) => {
return (
<p key={index}>{value.first_name} {value.last_name}</p>
)
})}
Your code you should now look something like this
import React from 'react';
import ReactDOM from 'react-dom';
import axios from 'axios';
var App = React.createClass({
getInitialState() {
return {
first_name: '',
last_name: '',
people: []
}
},
render() {
return (
<div>
<h1>Create Person</h1>
<input onChange={this.first_nameCatcher} type="text" />
<input onChange={this.last_nameCatcher} type="text" />
<button onClick={this.createPerson}>Create Person</button>
<br></br>
<br></br>
<h1>Read People</h1>
<button onClick={this.readPeople}>Read People</button>
{this.state.people.map( (value, index) => {
return (
<p key={index}>{value.first_name} {value.last_name}</p>
)
})}
</div>
)
},
first_nameCatcher(event) {
this.setState({
first_name: event.target.value
})
},
last_nameCatcher(event) {
this.setState({
last_name: event.target.value
})
},
createPerson() {
var person = {first_name: this.state.first_name, last_name: this.state.last_name}
axios({
method: 'POST',
url: 'http://localhost:3000/api/add/person',
data: person
})
},
readPeople() {
axios({
method: 'GET',
url: 'http://localhost:3000/api/people'
}).then(r => {
this.setState({
people: r.data
})
});
}
})
ReactDOM.render(<App />, document.getElementById('app'));
#CRUD - Update Using the id we can update a person's information. Since we're going for the bare minimum, you'll have to know the user id you want to edit and input it into a field along with the first_name and last_name (Black Diamond: Click a user to get it's id, and update it with first name OR last name (Good luck!))
1 - Just like create person, add 3 input fields and a button using onChange={}
& onClick={}
<input onChange={this.edit_idCatcher} type="text" />
<input onChange={this.edit_firstCatcher} type="text" />
<input onChange={this.edit_lastCatcher} type="text" />
<button onClick={this.updatePerson}>Update Person</button>
2 - Create three new variables in getInitialState() {}
edit_id, edit_first, edit_last.
getInitialState() {
return {
first_name: '',
last_name: '',
people: [],
edit_id: '',
edit_first: '',
edit_last: ''
}
},
3 - Update edit_it, edit_first, and edit_last using onChange={}
edit_idCatcher(event) {
this.setState({
edit_id: event.target.value
})
},
edit_firstCatcher(event) {
this.setState({
edit_first: event.target.value
})
},
edit_lastCatcher(event) {
this.setState({
edit_last: event.target.value
})
},
4 - Fire the api call to edit that user with onClick={}
(Black Diamond: Update person list after updating a user)
updatePerson() {
var person = {
id: this.state.edit_id,
first_name: this.state.edit_first,
last_name: this.state.edit_last
}
axios({
method: 'PUT',
url: 'http://localhost:3000/api/update/person',
data: person
})
}
Your code should now look something like this
import React from 'react';
import ReactDOM from 'react-dom';
import axios from 'axios';
var App = React.createClass({
getInitialState() {
return {
first_name: '',
last_name: '',
people: [],
edit_id: '',
edit_first: '',
edit_last: ''
}
},
render() {
return (
<div>
<h1>Create Person</h1>
<input onChange={this.first_nameCatcher} type="text" />
<input onChange={this.last_nameCatcher} type="text" />
<button onClick={this.createPerson}>Create Person</button>
<br></br>
<br></br>
<h1>Read People</h1>
<button onClick={this.readPeople}>Read People</button>
{this.state.people.map( (value, index) => {
return (
<p key={index}>{value.first_name} {value.last_name}</p>
)
})}
<br></br>
<br></br>
<h1>Update Person</h1>
<input onChange={this.edit_idCatcher} type="text" />
<input onChange={this.edit_firstCatcher} type="text" />
<input onChange={this.edit_lastCatcher} type="text" />
<button onClick={this.updatePerson}>Update Person</button>
</div>
)
},
first_nameCatcher(event) {
this.setState({
first_name: event.target.value
})
},
last_nameCatcher(event) {
this.setState({
last_name: event.target.value
})
},
createPerson() {
var person = {first_name: this.state.first_name, last_name: this.state.last_name}
axios({
method: 'POST',
url: 'http://localhost:3000/api/add/person',
data: person
})
},
readPeople() {
axios({
method: 'GET',
url: 'http://localhost:3000/api/people'
}).then(r => {
this.setState({
people: r.data
})
});
},
edit_idCatcher(event) {
this.setState({
edit_id: event.target.value
})
},
edit_firstCatcher(event) {
this.setState({
edit_first: event.target.value
})
},
edit_lastCatcher(event) {
this.setState({
edit_last: event.target.value
})
},
updatePerson() {
var person = {
id: this.state.edit_id,
first_name: this.state.edit_first,
last_name: this.state.edit_last
}
axios({
method: 'PUT',
url: 'http://localhost:3000/api/update/person',
data: person
})
}
})
ReactDOM.render(<App />, document.getElementById('app'));
#CRUD - Delete Making a delete call is very similar to updating, but instead of 3 inputs, we only need one which is the id of the user we want to delete.
1 - Add 1 input field and a button using onChange={}
& onClick={}
<input onChange={this.delete_idCatcher} type="text" />
<button onClick={this.deletePerson}>Delete Person</button>
2 - Create one new variable in getInitialState() {}
delete_id.
getInitialState() {
return {
first_name: '',
last_name: '',
people: [],
edit_id: '',
edit_first: '',
edit_last: '',
delete_id: ''
}
},
3 - Update delete_id using onChange={}
delete_idCatcher(event) {
this.setState({
delete_id: event.target.value
})
}
4 - Fire the api call to delete that user with onClick={}
deletePerson() {
axios({
method: 'DELETE',
url: `http://localhost:3000/api/person/${this.state.deleteId}`
})
}
Your code should now look something like this
import React from 'react';
import ReactDOM from 'react-dom';
import axios from 'axios';
var App = React.createClass({
getInitialState() {
return {
first_name: '',
last_name: '',
people: [],
edit_id: '',
edit_first: '',
edit_last: '',
delete_id: ''
}
},
render() {
return (
<div>
<h1>Create Person</h1>
<input onChange={this.first_nameCatcher} type="text" />
<input onChange={this.last_nameCatcher} type="text" />
<button onClick={this.createPerson}>Create Person</button>
<br></br>
<br></br>
<h1>Read People</h1>
<button onClick={this.readPeople}>Read People</button>
{this.state.people.map( (value, index) => {
return (
<p key={index}>{value.first_name} {value.last_name}</p>
)
})}
<br></br>
<br></br>
<h1>Update Person</h1>
<input onChange={this.edit_idCatcher} type="text" />
<input onChange={this.edit_firstCatcher} type="text" />
<input onChange={this.edit_lastCatcher} type="text" />
<button onClick={this.updatePerson}>Update Person</button>
<br></br>
<br></br>
<h1>Delete Person</h1>
<input onChange={this.delete_idCatcher} type="text" />
<button onClick={this.deletePerson}>Delete Person</button>
</div>
)
},
first_nameCatcher(event) {
this.setState({
first_name: event.target.value
})
},
last_nameCatcher(event) {
this.setState({
last_name: event.target.value
})
},
createPerson() {
var person = {first_name: this.state.first_name, last_name: this.state.last_name}
axios({
method: 'POST',
url: 'http://localhost:3000/api/add/person',
data: person
})
},
readPeople() {
axios({
method: 'GET',
url: 'http://localhost:3000/api/people'
}).then(r => {
this.setState({
people: r.data
})
});
},
edit_idCatcher(event) {
this.setState({
edit_id: event.target.value
})
},
edit_firstCatcher(event) {
this.setState({
edit_first: event.target.value
})
},
edit_lastCatcher(event) {
this.setState({
edit_last: event.target.value
})
},
updatePerson() {
var person = {
id: this.state.edit_id,
first_name: this.state.edit_first,
last_name: this.state.edit_last
}
axios({
method: 'POST',
url: 'http://localhost:3000/api/update/person',
data: person
})
},
delete_idCatcher(event) {
this.setState({
delete_id: event.target.value
})
},
deletePerson() {
axios({
method: 'DELETE',
url: `http://localhost:3000/api/person/${this.state.deleteId}`
})
}
})
ReactDOM.render(<App />, document.getElementById('app'));
#Congratulations You now know the basics of CRUD with React!